Vue3项目练习详细步骤(第三部分:文章分类页面模块)

文章分类列表

主体结构 

接口文档

文章分类列表查询接口数据绑定

 Pinia状态管理库

axios请求拦截器 

Pinia持久化插件-persist 

未登录统一处理

添加文章分类 

主体结构 

 接口文档

 绑定请求数据

编辑文章分类

弹框结构 

 数据回显

 接口文档

绑定请求数据

删除分类 

接口文档

 绑定请求数据

 

文章分类列表

主体结构 

 在ArticleCategory.vue文件中完成文章分类列表组件

<script setup>
import {
    Edit,
    Delete
} from '@element-plus/icons-vue'
import { ref } from 'vue'
const categorys = ref([
    {
        "id": 3,
        "categoryName": "美食",
        "categoryAlias": "my",
        "createTime": "2023-09-02 12:06:59",
        "updateTime": "2023-09-02 12:06:59"
    },
    {
        "id": 4,
        "categoryName": "娱乐",
        "categoryAlias": "yl",
        "createTime": "2023-09-02 12:08:16",
        "updateTime": "2023-09-02 12:08:16"
    },
    {
        "id": 5,
        "categoryName": "军事",
        "categoryAlias": "js",
        "createTime": "2023-09-02 12:08:33",
        "updateTime": "2023-09-02 12:08:33"
    }
])
</script>
<template>
    <el-card class="page-container">
        <template #header>
            <div class="header">
                <span>文章分类</span>
                <div class="extra">
                    <el-button type="primary">添加分类</el-button>
                </div>
            </div>
        </template>
        <el-table :data="categorys" style="width: 100%">
            <el-table-column label="序号" width="100" type="index"> </el-table-column>
            <el-table-column label="分类名称" prop="categoryName"></el-table-column>
            <el-table-column label="分类别名" prop="categoryAlias"></el-table-column>
            <el-table-column label="操作" width="100">
                <template #default="{ row }">
                    <el-button :icon="Edit" circle plain type="primary" ></el-button>
                    <el-button :icon="Delete" circle plain type="danger"></el-button>
                </template>
            </el-table-column>
            <template #empty>
                <el-empty description="没有数据" />
            </template>
        </el-table>
    </el-card>
</template>

<style lang="scss" scoped>
.page-container {
    min-height: 100%;
    box-sizing: border-box;

    .header {
        display: flex;
        align-items: center;
        justify-content: space-between;
    }
}
</style>

保存查看

接口文档

详细接口文档链接:https://blog.csdn.net/dafsq/article/details/138500082?spm=1001.2014.3001.5502 

文章分类列表查询接口数据绑定

在api目录下新建article.js文件

完成分类列表查询请求方法 

//导入请求工具文件
import request from '@/utils/request.js'

//文章分类列表查询
export const ArticleCategoryListService = () => {
    return request.get('/category')
}

 在文章分类ArticleCategory.vue文件中声明文章分类列表查询异步函数

//声明文章分类列表查询异步函数
import {articleCategoryListService} from '@/api/article.js'
const articleCategoryList = async() => {
    let result = await articleCategoryListService();
    if(result.code == 0){
        //成功获取
        categorys.value = result.data;
    }else{
        //获取失败
        ElMessage.error('获取失败')
    }
}
//调用
articleCategoryList();

 Pinia状态管理库

 Pinia是Vue的专属状态管理库,它允许你跨组件或页面共享状态

在项目目录下安装pinia

npm install pinia

在main.js文件中应用实例中使用pinia

创建src/stores/token.js文件 在其中定义store

在登录页面Login.vue文件中定义tokenStore并把登录时的token存入pinia

axios请求拦截器 

        当进入主页后,将来要与后台交互,都需要携带token,如果每次请求都写这样的代码,将会比较繁琐,此时可以将携带token的代码通过请求拦截器统一处理

在请求工具request.js文件中使用axios请求拦截器将token以请求头的方式携带请求

//导入token状态
import { useTokenStore } from '@/stores/token.js';
//添加请求拦截器
instance.interceptors.request.use(
    (config)=>{
        //在发送请求之前做什么
        let tokenStore = useTokenStore()
        //如果token中有值,在携带
        if(tokenStore.token){
            config.headers.Authorization=tokenStore.token
        }
        return config
    },
    (err)=>{
        //如果请求错误做什么
        Promise.reject(err)
    }
)

 保存可以查看到账号中已有的文章分类信息

Pinia持久化插件-persist 

  • Pinia默认是内存存储,当刷新浏览器的时候会丢失数据。
  • Persist插件可以将pinia中的数据持久化的存储 

在项目目录下安装persist

npm install pinia-persistedstate-plugin

在main.js文件在pinia中使用persist

在token.js文件中定义状态Store时指定持久化配置参数,在defineStore函数中添加第三个参数

 

这样登录刷新后token持久保存 就不会报错了 

 

未登录统一处理

        在后续访问接口时,如果没有登录,则前端不携带token,后台服务器会返回响应状态码401,代表未登录,此时可以在axios的响应拦截器中,统一对未登录的情况做处理

//导入Element-Plus提示框组件
import { ElMessage } from 'element-plus'

//导入路由
import router from '@/router/router.js'
//添加响应拦截器
instance.interceptors.response.use(
    result=>{
        return result.data;
    },
    err=>{
        //如果响应状态码时401,代表未登录,给出对应的提示,并跳转到登录页
        if(err.response.status===401){
            ElMessage.error('请先登录!')
            router.push('/login')
        }else{
            ElMessage.error('服务异常');
        }
        return Promise.reject(err);//异步的状态转化成失败的状态
    }
)

 

添加文章分类 

主体结构 

在ArticleCategory.vue文件中添加分类弹窗组件

                <!-- 添加分类弹窗 -->
        <el-dialog v-model="dialogVisible" title="添加弹层" width="30%">
            <el-form :model="categoryModel" :rules="rules" label-width="100px" style="padding-right: 30px">
                <el-form-item label="分类名称" prop="categoryName">
                    <el-input v-model="categoryModel.categoryName" minlength="1" maxlength="10"></el-input>
                </el-form-item>
                <el-form-item label="分类别名" prop="categoryAlias">
                    <el-input v-model="categoryModel.categoryAlias" minlength="1" maxlength="15"></el-input>
                </el-form-item>
            </el-form>
            <template #footer>
                <span class="dialog-footer">
                    <el-button @click="dialogVisible = false">取消</el-button>
                    <el-button type="primary"> 确认 </el-button>
                </span>
            </template>
        </el-dialog>

 添加表单校验和数据模型

//控制添加分类弹窗
const dialogVisible = ref(false)

//添加分类数据模型
const categoryModel = ref({
    categoryName: '',
    categoryAlias: ''
})
//添加分类表单校验
const rules = {
    categoryName: [
        { required: true, message: '请输入分类名称', trigger: 'blur' },
    ],
    categoryAlias: [
        { required: true, message: '请输入分类别名', trigger: 'blur' },
    ]
}

在添加分类按钮处绑定显示弹窗的单击事件

<el-button type="primary" @click="dialogVisible = true">添加分类</el-button>

点击添加文章按钮时弹窗正常显示

 绑定数据模型

绑定校验规则

已绑定成功

 绑定取消按钮事件,使弹框不显示

 接口文档

 绑定请求数据

在article.js文件中添加请求函数

//文章分类添加
export const articleCategoryAddService = (categoryModel) => {
    return request.post('/category',categoryModel)
}

在文章分类页面ArticleCategory.vue文件中添加单击事件请求函数

//导入Element-Plus提示框组件
import { ElMessage } from 'element-plus'
//导入articleCategoryAddService函数
import {articleCategoryAddService} from '@/api/article.js'
const addCategory = async () => {
    let result = await articleCategoryAddService(categoryModel.value);
    if(result.code == 0){
        //成功添加
        ElMessage(result.message? result.message:'添加成功')
        //隐藏弹窗
        dialogVisible.value = false
        //再次访问后台接口,查询所有分类
        articleCategoryList();
    }else{
        //添加失败
        ElMessage.error('添加失败')
    }
}

在确认按钮中绑定该事件函数 

<el-button type="primary" @click="addCategory"> 确认 </el-button>

 

编辑文章分类

弹框结构 

 在编辑按钮处添加事件,点击后显示弹框

<el-button :icon="Edit" circle plain type="primary" @click="dialogVisible = true"></el-button>

定义变量控制弹窗标题 

//定义变量控制弹窗标题
const title=ref('')

在弹窗标题中绑定变量

<el-dialog v-model="dialogVisible" :title="title" width="30%">

 在添加分类按钮中赋值给标题变量

<el-button type="primary" @click="dialogVisible = true;title='添加分类'">添加分类</el-button>

在编辑按钮中赋值标题变量

<el-button :icon="Edit" circle plain type="primary" @click="dialogVisible = true;title='编辑分类'"></el-button>

 保存之后点击不同的按钮将会显示对应的弹框标题

 数据回显

 定义数据回显函数

//修改分类回显
const updateCategoryEcho = (row) => {
    title.value = '修改分类'
    dialogVisible.value = true
    //将row中的数据赋值给categoryModel
    categoryModel.value.categoryName=row.categoryName
    categoryModel.value.categoryAlias=row.categoryAlias
    //修改的时候必须传递分类的id,所以扩展一个id属性
    categoryModel.value.id=row.id
}

 通过插槽的方式得到被点击按钮所在行的数据

    <template #default="{ row }">
        <el-button :icon="Edit" circle plain type="primary" @click="updateCategoryEcho(row)"></el-button>
        <el-button :icon="Delete" circle plain type="danger"></el-button>
    </template>

保存后点击编辑 文本框回显效果

 

 接口文档

 

绑定请求数据

在article.js文件中添加请求函数

 在文章分类页面ArticleCategory.vue文件中添加修改文章分类单击事件请求函数

//导入articleCategoryUpdateService函数
import {articleCategoryUpdateService} from '@/api/article.js'
//修改分类
const updateCategory=async ()=>{
        let result = await articleCategoryUpdateService(categoryModel.value)
        if(result.code == 0){
            ElMessage.success(result.message? result.message:'修改成功')
            //隐藏弹窗
            dialogVisible.value=false
            //再次访问后台接口,查询所有分类
            articleCategoryList();
    }else{
        //添加失败
        ElMessage.error('修改失败')
    }
}

由于现在修改和新增共用了一个数据模型,所以在点击添加分类后,有时候会显示数据,此时可以将categoryModel中的数据清空  

//清空模型数据
const clearData = ()=>{
    categoryModel.value.categoryName='',
    categoryModel.value.categoryAlias=''
}

 修改确定按钮的绑定事件

<el-button type="primary" @click="title==='添加分类'? addCategory():updateCategory()"> 确认 </el-button>

 修改添加按钮的点击事件

<el-button type="primary" @click="dialogVisible = true;title='添加分类';clearData()">添加分类</el-button>

 保存之后就能完成添加和修改的功能了

删除分类 

接口文档

 

 绑定请求数据

 在article.js文件中添加请求函数

//删除分类
export const articleCategoryDeleteService = (id) => {
    return request.delete('/category?id='+id)
}

  在文章分类页面ArticleCategory.vue文件中编写删除文章分类单击事件请求函数并在函数内部添加提示框组件

//导入element的ElMessageBox提示框组件
import { ElMessageBox } from 'element-plus'
//导入articleCategoryDeleteService函数
import {articleCategoryDeleteService} from '@/api/article.js'
//删除分类
const deleteCategory = (row) => {
    ElMessageBox.confirm(
        '确认是否删除该分类信息?',
        '提示',
        {
            confirmButtonText: '确认',
            cancelButtonText: '取消',
            type: 'warning',
        }
    )
        .then(async () => {
            //用户点击了确认
            let result = await articleCategoryDeleteService(row.id)
            ElMessage.success(result.message?result.message:'删除成功')
            //再次调用getAllCategory,获取所有文章分类
            articleCategoryList();
        })
        .catch(() => {
            //用户点击了取消
            ElMessage({
                type: 'info',
                message: '取消删除',
            })
        })
}

 在删除图标按钮处绑定该点击事件函数

<el-button :icon="Delete" circle plain type="danger" @click="deleteCategory(row)"></el-button>

 保存后即可正常删除文章分类

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

WAF几种代理模式详解

WAF简介 WAF的具体作用就是检测web应用中特定的应用&#xff0c;针对web应用的漏洞进行安全防护&#xff0c;阻止如SQL注入&#xff0c;XSS&#xff0c;跨脚本网站攻击等 正向代理 WAF和客户端与网络资源服务器都建立连接&#xff0c;但是WAF 的工作口具有自己的 IP 地址&…

【408真题】2009-28

“接”是针对题目进行必要的分析&#xff0c;比较简略&#xff1b; “化”是对题目中所涉及到的知识点进行详细解释&#xff1b; “发”是对此题型的解题套路总结&#xff0c;并结合历年真题或者典型例题进行运用。 涉及到的知识全部来源于王道各科教材&#xff08;2025版&…

智慧排水监测系统方案

智慧排水监测系统方案 智慧排水监测系统作为现代城市基础设施管理的重要组成部分&#xff0c;旨在通过先进的信息技术手段&#xff0c;实现对城市排水系统的全面、实时、高效的远程监控与管理。该系统整合了物联网技术、大数据分析、云计算平台与人工智能算法&#xff0c;不仅…

国产操作系统上apt命令详解 _ 统信 _ 麒麟 _ 中科方德

原文链接&#xff1a;国产操作系统上apt命令详解 | 统信 | 麒麟 | 中科方德 Hello&#xff0c;大家好啊&#xff01;今天给大家带来一篇在国产操作系统上使用apt命令的详解文章。apt&#xff08;Advanced Package Tool&#xff09;是Debian及其衍生发行版&#xff08;如统信UOS…

Adobe Camera Raw 11 for Mac/win:摄影后期处理的革命性飞跃

在数字摄影的世界里&#xff0c;RAW格式以其未压缩的原始数据特性&#xff0c;为摄影师提供了更大的后期处理空间。而Adobe Camera Raw 11&#xff0c;作为这一领域的翘楚&#xff0c;以其卓越的性能和创新的功能&#xff0c;为摄影师们带来了前所未有的创作体验。 Adobe Came…

深度学习(一)

深度学习&#xff08;一&#xff09; 一、实验目的 掌握前馈全连接神经网络&#xff0c;具体包括&#xff1a; (1) 前馈全连接神经网络的网络结构 (2) 前馈神全连接经网络的工作原理 (3) 前馈全连接神经网络的代码实现 二、实验内容 1. 导入常用工具包 2. 数据导入与数据…

国内加密软件排行榜,每一款加密软件都是精品

在数字化快速发展的今天&#xff0c;数据安全和隐私保护已成为企业和个人关注的焦点。加密软件作为保护数据安全的重要手段&#xff0c;其重要性日益凸显。以下是根据权威机构的评测和用户反馈&#xff0c;整理的国内加密软件排行榜及其特点概述。 1、加密软件安企神免费试用7天…

计算机系统基础实验三(解了但尽量理解)

一.准备阶段 1、下载好32位的实验代码后&#xff0c;将文件解压缩并且通过共享文件夹操作将文件添加到虚拟机中&#xff0c;双击查看bomb.c代码&#xff0c;将c代码完整看了一遍&#xff0c;发现看这里的c代码是无从下手的&#xff0c;代码中只含有主函数&#xff0c;触发炸弹…

前缀和(下)

目录 热身&#xff1a; 寻找数组的中心下标 题解&#xff1a; 代码&#xff1a; 进阶&#xff1a; 除自身之外数组的乘积 题解&#xff1a; 代码&#xff1a; 和为K的子数组 题解&#xff1a; 代码&#xff1a; 和可被 K 整除的子数组 题解&#xff1a; 同余定理…

【网络运维的重要性】

&#x1f308;个人主页: 程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

基于单片机的步进电机控制系统研究

摘 要 &#xff1a; 近年来 &#xff0c; 步进电机凭借其定位精度高 、 使用方便 、 性价比高 、 容易控制等优点 &#xff0c; 在各领域受到广泛应用 。 文中利用C52 单片机设计了一种步进电机控制系统 &#xff0c; 介绍了其总体方案 、 主控制模块 、 驱动电路 、 键盘 、 晶…

jmeter多用户并发登录教程

有时候为了模拟更真实的场景&#xff0c;在项目中需要多用户登录操作&#xff0c;大致参考如下 jmx脚本&#xff1a;百度网盘链接 提取码&#xff1a;0000 一&#xff1a; 单用户登录 先使用1个用户登录&#xff08;先把1个请求调试通过&#xff09; 发送一个登录请求&…

【Python】解决Python报错:TypeError: ‘int‘ object is not iterable

&#x1f9d1; 博主简介&#xff1a;阿里巴巴嵌入式技术专家&#xff0c;深耕嵌入式人工智能领域&#xff0c;具备多年的嵌入式硬件产品研发管理经验。 &#x1f4d2; 博客介绍&#xff1a;分享嵌入式开发领域的相关知识、经验、思考和感悟&#xff0c;欢迎关注。提供嵌入式方向…

打造云计算时代的仿真软件

2024年5月25日&#xff0c;北京云道智造科技有限公司&#xff08;下称“云道智造”&#xff09;在深圳成功举办了2024新品发布会暨用户大会。来自全国各地的近500位客户和合作伙伴代表齐聚一堂&#xff0c;共同见证了云道智造新产品的隆重发布&#xff0c;交流分享了仿真领域的…

模型实战(21)之 C++ - tensorRT部署yolov8-det 目标检测

C++ - tensorRT部署yolov8-det 目标检测 python环境下如何直接调用推理模型转换并导出:pt -> onnx ->.engineC++ tensorrt 部署检测模型不写废话了,直接上具体实现过程+all代码 1.Python环境下推理 直接命令行推理,巨简单yolo detect predict model=yolov8n.pt source…

SQL刷题笔记day6——转战LeetCode

1 第二高的薪水 ​ 我的代码&#xff1a; SELECT Salary SecondHighestSalary FROM Employee ORDER BY Salary DESC LIMIT 1, 1 我的代码不满足示例2的情况&#xff1a;如果没有第 2 高的薪资&#xff0c;即表里可能只有一条记录&#xff0c;这个解答会被评测为 Wrong Answ…

构造+模拟,CF1148C. Crazy Diamond

一、题目 1、题目描述 2、输入输出 2.1输入 2.2输出 3、原题链接 Problem - 1148C - Codeforces 二、解题报告 1、思路分析 题目提示O(5n)的解法了&#xff0c;事实上我们O(3n)就能解决&#xff0c;关键在于1&#xff0c;n的处理 我们读入数据a[]&#xff0c;代表初始数组…

模型实战(22)之 C++ - tensorRT部署yolov8-cls 目标分类

C++ - tensorRT部署yolov8-cls 目标分类 在检测应用场景中如果有同等类别不同形态的目标,单纯的目标检测可能达不到实用或者想要的精度,这就需要衔接一步分类python环境下如何直接调用推理模型转换并导出:pt -> onnx ->.engineC++ tensorrt 部署分类模型1.Python环境下…

WWW24因果论文(1/8) | 利用强化学习(智能体)进行因果问答

【摘要】因果问题询问不同事件或现象之间的因果关系。它们对于各种用例都很重要&#xff0c;包括虚拟助手和搜索引擎。然而&#xff0c;许多当前的因果问答方法无法为其答案提供解释或证据。因此&#xff0c;在本文中&#xff0c;我们旨在使用因果关系图来回答因果问题&#xf…

【OrangePi AIpro】开箱初体验以及OAK深度相机测试

1. 简介 Orangepi AIPRO 是一款采用昇腾AI技术路线&#xff0c;集成4核64位处理器AI处理器的单板计算机&#xff0c;集成图形处理器&#xff0c;支持8TOPS AI算力&#xff0c;拥有8GB/16GB LPDDR4X&#xff0c;可以外接eMMC模块&#xff0c;支持双4K高清输出。 Orange Pi AIpr…