vue2文章添加多个标签思路代码及效果展示

效果展示

tags

思路

data数据结构

第一个数组,用来存放标签库,供创建文章时选择

第二个数组,用来存放从标签库选中后的标签, 且选中后需在可选的标签库里删除,否则出现同一个标签被多次添加

js代码

点击输入框,可展示所有标签/也可不展示,取决于组件身上属性(激活时展示or不展示)

点击想要的标签,标签库移除,添加到选中数组内

移除标签,标签库添加回去(后续还想选),然后从选中数组内移除

是不是听的云里雾里?这里放一段gif演示

tags1

确定的时候就把选中标签数组提交到后台这没什么好说的,数组转字符串扔过去即可

后端数据库使用String=>varchar类型存储

不过这里设计成弹窗式的,当用户取消之后需要清空选中数组,标签库重置

废话不多说了,上代码

步骤

前端

前端搜索时展示数据

使用element的autocomplete组件

组件

<el-autocompletev-model="keyWord"class="inline-input":fetch-suggestions="querySearch"placeholder="请输入内容"style="margin-right: 10px"@select="handleSelect"/>

data

image-20240725163016450

method

搜索时回显数据

//用于like查询 
querySearch(queryString, cb) {var tags = this.tagsvar results = queryString ? tags.filter((tag) => {return (tag.value.toLowerCase().indexOf(queryString.toLowerCase()) >= 0)}) : tags// 调用 callback 返回建议列表的数据cb(results)},

点击时加入已选中标签数组

    handleSelect(item) {this.keyWord = ''// 加入选中组this.selectedTags.push(item)// 选中后该标签不可被再选中,从标签库移除this.tags = this.tags.filter((i) => i.value !== item.value)},

关闭标签时的移除&加入各自的数组

handleCloseTag(tag) {// 还给标签库this.tags.push(tag)// 移除取消的标签this.selectedTags = this.selectedTags.filter((item) => item.value !== tag.value)},

挂载时回调获取数据

 async selectAllTags() {const res = await selectAllTag(1, 99)if (res.code === 20000) {this.tags = res.data.records// 由于element的autocomplete 搜索时回显的数据字段必须叫value,这里给他重新包装一下this.tags = this.tags.map((tag) => {return {value: tag.tagName}})//   .filter((item) => {// 由于后续编辑回显selectedTag也有值,产生关闭后,unselect加值,重复//   // 与已选中数组一致的不会进  可选列表//   // 这里的some在返回值为true时直接跳出, 唯一为真,如果是every则要么找到一个false要么全为true才结束//   return !this.selectedTags.some(selected => selected.value === item.value)// })this.unselectedTags = this.tags} else {console.log('服务器废了')}},

这里注释掉的是为了解决编辑回显时 选中数组和标签库产生重复问题,你可以自行研究完添加后在将其注释解开慢慢看

保存提交数据

  this.tags = this.selectedTags.map((item) => {return item.value})this.form.tags = this.tags.join(',')

页面完整代码

<template><div v-if="initSuccess" class="main"><div class="operate" style="display: flex;justify-content: space-between"><div style="display: flex;"><el-input v-model="form.title" placeholder="标题" /><el-select v-model="form.categoryName" placeholder="请选择" style="margin:0 20px;min-width: 150px"><el-optionv-for="category in categoryData":key="category.categoryName":label="category.categoryName":value="category.categoryName"/></el-select><el-input v-model="form.summary" placeholder="摘要" style="margin-right: 20px" /><el-button style="margin-right: 10px;" @click=" handleAdd">添加标签 +</el-button><el-dialog title="添加标签" :visible.sync="dialogVis" style="margin-top: -100px"><!--          <el-input v-model="keyWord" @input="searchTag" />--><el-autocompletev-model="keyWord"class="inline-input":fetch-suggestions="querySearch"placeholder="请输入内容"style="margin-right: 10px"@select="handleSelect"/><el-tagv-for="(tag ,index) in selectedTags":key="index"class="singleTag"style="margin-right: 10px"closable@close="handleCloseTag(tag)">{{ tag.value }}</el-tag><div slot="footer" class="dialog-footer"><el-button @click="cancelAddTag">重 置</el-button><el-button type="primary" @click="dialogVis=false">确 定</el-button></div></el-dialog><el-tagv-for="(tag ,index) in selectedTags":key="index"class="singleTag"style="margin-right: 10px"closable@close="handleCloseTag(tag)">{{ tag.value }}</el-tag></div><div style="display: flex"><div class="publishBox" @click="save">保存</div><div class="cancelBox" @click="$router.push('/article/list')">取消</div></div></div><MdEditor :content="form.content" @update:content="getEditorContent" /></div>
</template><script>import MdEditor from '@/components/MdEditor/index.vue'
import { add, update } from '@/api/article/list'
import { selectList as selectCategoryList } from '@/api/article/category'
import { selectList as selectAllTag } from '@/api/tag'export default {name: 'Index',components: { MdEditor },data() {return {articleId: this.$route.params.id,article: {},content: '',form: {},categoryData: [],initSuccess: false,dialogVis: false,keyWord: '',selectedTags: [],unselectedTags: [],tags: []}},mounted() {if (this.$route.params.id !== '0') {this.form = this.$store.state.currArticlethis.selectedTags = this.form.tags.split(',').map(item => {return {value: item}})}selectCategoryList(1, 999, this.searchObj).then(res => {this.categoryData = res.data.recordsthis.listLoading = falsethis.initSuccess = true})this.selectAllTags()},methods: {cancelAddTag() {this.keyWord = ''this.selectedTags = []this.tags = this.unselectedTagsthis.dialogVis = false},handleCloseTag(tag) {// 还给标签库this.tags.push(tag)// 移除取消的标签this.selectedTags = this.selectedTags.filter((item) => item.value !== tag.value)},querySearch(queryString, cb) {var tags = this.tagsvar results = queryString ? tags.filter((tag) => {return (tag.value.toLowerCase().indexOf(queryString.toLowerCase()) >= 0)}) : tags// 调用 callback 返回建议列表的数据cb(results)},handleSelect(item) {this.keyWord = ''// 加入选中组this.selectedTags.push(item)// 选中后该标签不可被再选中,从标签库移除this.tags = this.tags.filter((i) => i.value !== item.value)},async selectAllTags() {const res = await selectAllTag(1, 99)if (res.code === 20000) {this.tags = res.data.records// 由于element的autocomplete 搜索时回显的数据字段必须叫value,这里给他重新包装一下this.tags = this.tags.map((tag) => {return {value: tag.tagName}})//   .filter((item) => {// 由于后续编辑回显selectedTag也有值,产生关闭后,unselect加值,重复//   // 与已选中数组一致的不会进  可选列表//   // 这里的some在返回值为true时直接跳出, 唯一为真,如果是every则要么找到一个false要么全为true才结束//   return !this.selectedTags.some(selected => selected.value === item.value)// })this.unselectedTags = this.tags} else {console.log('服务器废了')}},handleAdd() {this.dialogVis = true},getEditorContent(content) {this.form.content = content},save() {// 将原来的对象数组转为数组,提取出对象的value值this.tags = this.selectedTags.map((item) => {return item.value})this.form.tags = this.tags.join(',')if (this.articleId === '0') {add(this.form).then(res => {if (res.code === 20000) {this.$message.success('保存成功')this.$router.push('/article/list')} else {this.$message.error(res.msg)}})} else {update(this.form).then(res => {if (res.code === 20000) {this.$message.success('保存成功')this.$router.push('/article/list')} else {this.$message.error(res.msg)}})}}}
}
</script><style scoped>
.singleTag {position: relative;
}.del {position: absolute;font-size: 12px;color: red;cursor: pointer;display: block;/*background: red;*/bottom: 0;right: -10px;width: 20px;text-align: center;transition: 1s;
}.del:hover {color: #2f4d03;cursor: pointer;transform: rotateZ(360deg);
}.publishBox {cursor: pointer;width: 100px;text-align: center;margin-right: 20px;padding: 10px;background: #6ce8ff;color: #000000;box-shadow: 0 0 4px black;border-radius: 10px;
}.cancelBox {cursor: pointer;width: 100px;text-align: center;padding: 10px;background: #ff8383;color: #000000;box-shadow: 0 0 4px black;border-radius: 10px;
}
</style>

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

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

相关文章

智能APK动态防护系统:自动重命名与签名,实现安全分发

本智能APK动态防护系统通过集成先进的自动化处理技术&#xff0c;实现了对APK文件的深度定制化与安全性强化。系统核心功能包括自动反编译APK、随机生成包名与签名、代码混淆等&#xff0c;最终回编译生成独一无二的APK安装包。这一过程每5分钟&#xff08;时间间隔可自定义&am…

Windows下ORACLE数据泵expdp和impdp使用

Windows下ORACLE数据泵expdp和impdp使用 一、基础环境 操作系统&#xff1a;Windows server 2008&#xff1b; 数据库版本&#xff1a;Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production 数据库工具&#xff1a;PL/SQL 12.0.7 实验内容&…

示例:WPF中如何处理TabControl页面绑定ItemsSource切换TabItem时UI数据没有持久保存的问题

一、目的&#xff1a;在WPF开发过程中&#xff0c;经常用到TabControl&#xff0c;也会遇到类似问题&#xff0c;用TabControl绑定数据源ItemsSource时&#xff0c;切换TabItem时&#xff0c;UI上的数据没有持久保存&#xff0c;本文介绍一种处理方式&#xff0c;可以做到缓存页…

什么是云服务器ecs,为什么要选择云服务器

云服务器 ECS&#xff08;Elastic Compute Service&#xff09;是阿里云&#xff08;Alibaba Cloud&#xff09;提供的一种基于云计算的虚拟服务器服务。它允许用户在云端虚拟化环境中配置和管理服务器&#xff0c;无需投资物理硬件、提高资源利用率、降低维护成本、实现快速部…

Sed工具

文章目录 一、sed是什么二、sed的常用操作选项三、如何使用sed1.Sed结合正则表达式输出指定行2.增加内容3.删除4.替换5.搜索替换6.插入文件7.另存为到文件8.同时编辑9.分组操作10.读取完退出11.sed脚本12.sed的高级应用 一、sed是什么 sed 命令是利用脚本来处理文本文件。它可…

Redis的集群的搭建

1、为什么要搭建Redis集群 Redis 集群能够提供高可用性、高性能、扩展性和数据安全性&#xff0c;适用于各种需要高速缓存和数据存储的复杂应用场景 2、Redis的集群模式 主从模式哨兵模式区中心化模式 3、主从模式 redis主从模式表示一个主节点跟若干个从节点。主节点可以…

VMware 上安装 CentOS 7 教程 (包含网络设置)

**建议先看一些我安装VMware的教程&#xff0c;有些网络配置需要做一下 1.打开VMware&#xff0c;创建虚拟机 2.勾选自定义&#xff0c;点击下一步 3.点击下一步 4.勾选“稍后安装操作系统”&#xff0c;点击下一步 5.勾选linux&#xff0c;勾选centos7&#xff0c;点击下一步…

AH1405芯片的应用领域有哪些?sot23-5封装ic

1405芯片是一种SOT23-5封装的降压转换器&#xff0c;以其出色的性能和广泛的应用领域&#xff0c;成为电子设计中的热门选择。本文将详细介绍1405芯片的技术特点以及其在不同领域的应用情况。 技术特点 1. 宽输入电压范围 1405芯片能够接受从6V至40V的输入电压&#xff0c;这…

汽车绝缘检测详细设计

粘连检测原理 粘连检测&#xff1a; 目的&#xff1a;检测继电器、开关或电气触点是否因故障而保持在接通或断开的状态。工作原理&#xff1a; 正常操作&#xff1a;继电器或开关在正常操作时会周期性地开闭。开闭过程中会有明显的电流和电压变化。粘连状态&#xff1a;如果继…

Vuex数据持久化实现

版本&#xff1a;vue 3.4.29 vuex4.1.0 1. 出现的问题 当我使用 vuex 作为状态管理组件来存储用户的一些信息之后&#xff0c;发现从/login 页面跳转到/home 界面后拿不到vuex信息。 之后查阅资料了解&#xff0c;当切换路由后&#xff0c;vue 会重新渲染&#xff0c;而vuex 也…

pgsql的update语句在set里进行字段的运算 SET sort = sort +1

一、场景 需求&#xff1a;version 版本字段是记录数据更新的次数&#xff0c;新增时自动填充 version1 ,每更新一次数据 version就自增1。项目里单表插入和更新要手写update语句进行插入和更新。 –表中int4类型的字段 version 是1时&#xff0c;由1变成2 – version 是null…

【Linux】信号(signal)

目录 一、信号概念&#xff1a; 二、信号的常见状态&#xff1a; 信号递达&#xff1a; 信号未决&#xff1a; 阻塞信号&#xff1a; 忽略信号&#xff1a; 信号在内核中的表示&#xff1a; 三、信号相关函数&#xff1a; sigset_t &#xff08;类型&#xff09;&…

二、QGroundControl开发环境搭建

文章目录 环境列表QGC源码下载编译 环境列表 QGC GithubPX4-AutopilotQt 5.15Ubuntu20.04 QGC源码下载编译 官网下载指令 如下 // Clone the repo (or your fork) including submodules: git clone --recursive -j8 https://github.com/mavlink/qgroundcontrol.git // Upda…

Axure中继器实战篇:让数据展示和交互设计更上一层楼!

Axure中继器实战篇:让数据展示和交互设计更上一层楼! 前言 经过了前两章的学习,接下来我们去模拟的实际场景开启实战篇,以下是界面 1.前期准备 前期把页面准备好后,给中继器的每个单元格命名为了方便数据绑定的操作。 为了演示我准备了几十行数据,建议也多准备一点。…

后端返回一个图片链接,前端如何实现下载功能?

纯原创文章&#xff0c;转载请说明来源。 一、背景 要实现一个下载功能&#xff0c;后端直接返回了一个图片的地址https://xxxxx/pic.jpg。如果我们直接通过window.open(url, _blank) 的方式去下载这个图片&#xff0c;会发现 Chrome 浏览器会对这个图片进行预览&#xff0c;…

魅族手机怎么录屏?详细步骤助你轻松上手

“有人知道魅族手机怎么录屏吗&#xff0c;最近我在准备一些教学视频&#xff0c;急需用到手机的录屏功能来记录操作过程&#xff0c;但遗憾的是&#xff0c;我翻遍了设置也没能找到录屏的开关。所以&#xff0c;我在这里想问问大家&#xff0c;魅族手机是如何启动录屏功能的&a…

【PyTorch】图像多分类项目部署

【PyTorch】图像多分类项目 【PyTorch】图像多分类项目部署 如果需要在独立于训练脚本的新脚本中部署模型&#xff0c;这种情况模型和权重在内存中不存在&#xff0c;因此需要构造一个模型类的对象&#xff0c;然后将存储的权重加载到模型中。 加载模型参数&#xff0c;验证模型…

图解 HDFS 架构 |读写过程

HDFS HDFS 全称 Hadoop Distributed File System&#xff0c;是一个分布式文件系统。HDFS&#xff08;Hadoop Distributed File System&#xff09;是 Apache Hadoop 生态系统的一部分&#xff0c;它是一个分布式文件系统&#xff0c;用于存储和处理大规模数据集。HDFS 专门设…

源代码防泄密如何做?企业如何有效选择源代码防泄密产品?

源代码防泄密怎么选&#xff1f;如何高效做源代码防泄密工作&#xff1f; 源代码开发环境复杂&#xff0c;涉及的开发软件和文件类型众多且变化多端&#xff0c;那么究竟有哪些源代码防泄密软件能够适应各种开发软件而不影响原有的工作效率呢&#xff1f; 对于研发人员来说&a…

探索 Framer Motion 高级动画技巧:提升前端设计水平

在现代的网页和应用设计中&#xff0c;动画不仅仅是视觉的点缀&#xff0c;更是用户体验的重要组成部分。它能够使界面更具吸引力&#xff0c;提升交互的流畅性&#xff0c;甚至在不经意间传达品牌的个性和态度。然而&#xff0c;要创造出令人惊叹的动效并不容易——直到有了 F…