vue封装组件(一)标签和下拉框组合实现添加数据

背景: 最近接入短剧内容,需要添加短剧合作方。在详情页需要支持添加组件

方案一:标签tag加上输入框实现添加数据。图片见下 这个是刚开始做的,后来产品觉得这样会造成随意修改数据,需要改成下拉框形式添加

方案二:标签tag加上下拉框以及添加按钮实现。 这个就避免了乱添加数据,添加合作方名字需要在一个管理页面添加,然后下拉框请求自然就请求到数据了

方案一效果图:第一行 流程:点击新增后,填入数据,后台会自动检查是否存在,存在返回数据,否则保存然后返回数据。 这样做有个问题,一不小心填错数据,后台也保存下来了

方案二效果图:第二行 流程:下拉框中是已经存在的数据,然后选择添加就行。合作方需要在另外的一个管理页面增删查改。

接下来是代码实现。 注意:这两个都是封装的组件,直接掉用即可,既然是封装的组件,那么就要支持已经存在的合作方数据填入。

方案一

<template>

 <div>
<!--   {{shortPlays}}-->
   <el-tag style="margin-right: 5px" v-for="(tag,index) in shortPlaysFilter"
           :key="index" closable
           @close="close(index,tag.shortPlayName)">
     {{tag.shortPlayName}}
   </el-tag>

   <el-input
       v-if="inputVisible"
       v-model="inputValue"
       ref="ref"
       class="input-new-tag"
       size="small"
       style="width: 100px"
       @keyup.enter.native="inputConfirm"
       @blur="inputConfirm"
   ></el-input>
   <el-button v-else size="small" style="margin-left: 5px;" @click="btnClick">新增合作方</el-button>
 </div>

</template>

<script>
import * as mapRequest from "@/api/map";

export default {
  name: "ag_album_short_play_partner",
  props:{//这里外部的数据传入
    shortPlays:{
      required:true,
      default:()=>{
        return [];
      }
    }
  },
  data() {
    return {
      inputVisible:false,
      inputValue:'',
    }
  },
  created() {
  },
  methods: {
    close(index,name){
      // 注意:这里的index 不能直接拿到删除 因为这个index 是过滤后的 所以通过name找下标 然后删除
      console.log("name",name);
      let findex=this.shortPlays.findIndex(f=>f.shortPlayName==name);
      console.log("findex",findex);
      if(findex==-1 || findex<0){
        this.$message.error("删除短剧合作方失败!请联系开发人员");
        return;
      }
      //console.log(findex);
      let data=this.shortPlays[findex];
      console.log("data",data);
      if(data.id){// 存在id 说明本身就有合作方 标记删除
        this.$set(data,'delFlag',true);
        this.$emit('input',this.shortPlays);
      }else{
        this.shortPlays.splice(findex,1);
        this.$emit('input',this.shortPlays);
      }
    },
    btnClick(){
      this.inputVisible=true;
      this.$nextTick(()=>{
        this.$refs.ref.$refs.input.focus();
      })
    },
    inputConfirm(){
      let inputValue=this.inputValue;
      if(inputValue){
        mapRequest.addShortPlay({name:inputValue}).then(res=>{
          let data=res.data.data;
          this.shortPlays.push({addFlag:true,shortPlayId:data.id,shortPlayName:data.name});
          this.$emit('input',this.shortPlays);
        }).catch(err=>{
          this.$message.error("添加短剧合作方失败");
          console.log(err);
        })
      }
      this.inputVisible=false;
      this.inputValue='';
    }

  },
  computed: {
    shortPlaysFilter(){
      let obj=this.shortPlays.filter(item=>{
        if(item.delFlag&&item.delFlag==true) return false;
        return true;
      });
      return obj;
    }
  },
  watch: {},
  components: {},

}
</script>

<style scoped>

</style>

这个就是封装的组件,然后想用的时候直接引用即可。

方案二

<!--
删除逻辑:
1、原始tags中有delFlag true 这种数据不能展示,所以用 filteredTags 过滤
2、如果是本来存在的删除时可以标记delFlag true,如果时添加后(没点击提交动作,数据就没保存)又删除,这种直接删除即可 因为数据库没保存这些数据

[ { "addFlag": true, "shortPlayId": 8, "shortPlayName": "北京攸合" } ]
-->
<template>
  <div>
<!--    {{tags}}-->
    <el-tag v-for="(tag,index) in filteredTags" :key="index" closable @close="handleClose(index)">
      {{tag.shortPlayName}}
    </el-tag>
    <el-select v-model="bindModel" filterable clearable value-key="id"  placeholder="请选择短剧合作方" size="small" style="margin-left: 5px;width: 150px">
      <el-option v-for="item in options" :key="item.id" :label="item.name" :value="item">
      </el-option>
    </el-select>
    <el-button type="primary" size="small" style="margin-left: 5px" @click="addClick">添加</el-button>
  </div>
</template>

<script>
import * as mapRequest from "@/api/map";
export default {
  name: "ag-album-short-play-partner-v2",
  props:{
    tags:{
      type:Array,
      required:true,
      default:()=>{
        return [];
      }
    }
  },
  data() {
    return {
      bindModel:{

      },
      options:[],
    }
  },
  created() {
    this.getList();
  },
  methods: {
    getList(){
      this.options=JSON.parse('[{"id":8,"name":"北京攸合","createdAt":"2023-12-26 14:26:09","updatedAt":"2023-12-26 14:26:09"},{"id":9,"name":"网易","createdAt":"2024-01-03 10:13:37","updatedAt":"2024-01-03 10:13:37"}]');
      //这里是要请求后台数据的
      // mapRequest.findShortPlayList({},{}).then(res=>{
      //   this.options=res.data.data.records;
      // }).catch(err=>{
      //   this.$message.error(err.data.message);
      // })
    },
    handleClose(index){
      const tag=this.filteredTags[index];
      // console.log("-----",tag);
      if(tag==null){
        this.$message.error("删除错误!");
        return;
      }
      const originalIndex = this.tags.findIndex(t => t === tag);
      if (originalIndex !== -1 && this.tags[originalIndex].id ) {
        this.$set(this.tags[originalIndex], 'delFlag', true);
        this.$emit('input', this.tags);
      }else if (originalIndex !=-1 && (this.tags[originalIndex].id === null || this.tags[originalIndex].id === undefined)  ){
        this.tags.splice(originalIndex,1);
      }
      // this.$set(tag,'delFlag',true);
      // console.log("data",tag,",tags:",this.tags);
      // this.$emit('input',this.tags);


    },
    addClick(){
      // 请勿重复添加
      if(this.bindModel.id){
        // console.log("@@@",this.bindModel);
        let data=this.bindModel;
        this.tags.push({addFlag:true,shortPlayId:data.id,shortPlayName:data.name});
        // console.log("---",this.tags);
        this.$emit('input',this.tags);
        this.bindModel=null;
        return;
      }else{
        this.$message.warning("请选择合作方后再添加~");
        return;
      }

    },
  },
  computed: {
    filteredTags() {
      return this.tags.filter(tag => !tag.delFlag);
    }
  },
  watch: {},
  components: {},

}
</script>

<style scoped>

</style>

最终就是外面的调用

<!--
用来写测试案例
-->
<template>
  <div>
    <ag-album-short-play-partner :short-plays="shortPlays"></ag-album-short-play-partner>
    <ag-album-short-play-partner-v2 :tags="shortPlays"></ag-album-short-play-partner-v2>
  </div>


</template>


<script>
import agAlbumShortPlayPartner from "@/views/longVideo/album/components/ag_album_short_play_partner";
import AgAlbumShortPlayPartnerV2 from "@/views/longVideo/album/components/ag_album_short_play_partner_v2";
export default {
  name:'videoDirUploadList',
  data(){
    return{
      data:{
        album:{

        }
      },
      shortPlays:[{"id":8,"albumId":44116442,"shortPlayId":9,"createdAt":"2024-01-04 15:50:59","updatedAt":"2024-01-04 15:50:59","shortPlayName":"网易","delFlag":"","addFlag":""}],

    }
  },
  created() {
  },
  methods:{
  

  },
  components:{
    AgAlbumShortPlayPartnerV2,
    agAlbumShortPlayPartner,
  }

}
</script>

<style>
</style>

ok,本期结束,关注作者不迷路~持续更新好用的组件~

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

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

相关文章

跟随chatgpt从零开始安装git(Windows系统)

为什么我们要安装Git&#xff1f;Git有什么用&#xff1f; 1. 版本控制&#xff1a;Git 可以追踪代码的所有变化&#xff0c;记录每个提交的差异&#xff0c;使您能够轻松地回溯到任何历史版本或比较不同版本之间的差异。 2. 分支管理&#xff1a;通过 Git 的分支功能&#xff…

【数据结构和算法】字符串解码

其他系列文章导航 Java基础合集数据结构与算法合集 设计模式合集 多线程合集 分布式合集 ES合集 文章目录 其他系列文章导航 文章目录 前言 一、题目描述 二、题解 2.1 什么情况会用到栈 2.2 方法一&#xff1a;辅助栈法 三、代码 3.1 方法一&#xff1a;辅助栈法 四…

UE5.1_UMG序列帧动画制作

UE5.1_UMG序列帧动画制作 UMG序列帧动画制作相对比较简单&#xff0c;不像视频帧需要创建媒体播放器那么复杂&#xff0c;以下简要说明&#xff1a; 1. 事件函数 2. 准备序列帧装入数组 3. 构造调用事件函数 4. 预览 序列帧UMG0105 5. 完成&#xff01;按需配置即可。

洗地机、扫地机器人和吸尘器哪个好?三选一谁更值得买?

传统的清洁地面方式&#xff0c;不仅耗费时间、精力&#xff0c;还会造成人的腰酸背痛&#xff0c;带来一连串的家务后遗症&#xff0c;简直是苦不堪言。像洗地机、扫地机器人、吸尘器等电动清洁工具的诞生让人们的清洁更加轻松省事&#xff0c;也凭借着这些优势深受大众喜爱。…

Python基础(十九、文件操作写入与追加)

文章目录 一、文件的写入&#xff08;使用 "w" 模式&#xff09;二、文件的追加&#xff08;使用 "a" 模式&#xff09;三、文件备份案例接之前的答案 在 Python 中&#xff0c;open() 是一个内置函数&#xff0c;用于打开文件并返回文件对象。它是处理文件…

2024更新阿里云域名优惠口令大全_优惠口令获取方法

2024年阿里云域名优惠口令&#xff0c;com域名续费优惠口令“com批量注册更享优惠”&#xff0c;cn域名续费优惠口令“cn注册多个价格更优”&#xff0c;cn域名注册优惠口令“互联网上的中国标识”&#xff0c;阿里云优惠口令是域名专属的优惠码&#xff0c;可用于域名注册、续…

Python学习之路——文件部分【文件的读取】

目录 先解释一下引文的答案 一、open()打开函数 二、mode常用的三种基础访问模式 三、读-操作相关方法 &#xff08;一&#xff09;read方法 &#xff08;二&#xff09;readlines方法 &#xff08;三&#xff09;with open 语法 &#xff08;四&#xff09;操作汇总 …

损失函数篇 | YOLOv8 引入 Shape-IoU 考虑边框形状与尺度的度量

作者导读&#xff1a;Shape-IoU&#xff1a;考虑边框形状与尺度的度量 论文地址&#xff1a;https://arxiv.org/abs/2312.17663 作者视频解读&#xff1a;https://www.bilibili.com 开源代码地址&#xff1a;https://github.com/malagoutou/Shape-IoU/blob/main/shapeiou.py…

代码随想录day21 二叉搜索树进阶

530.二叉搜索树的最小绝对差 题目 给你一棵所有节点为非负值的二叉搜索树&#xff0c;请你计算树中任意两节点的差的绝对值的最小值。 示例&#xff1a; 思考 本题有一种笨办法&#xff0c;就是把二叉树的所有结点都存到一个vector里&#xff0c;因为二叉搜索树是左中右排序…

Spring整合MyBatis框架!!!

搭建环境&#xff1a; pom.xml: <properties><maven.compiler.source>17</maven.compiler.source><maven.compiler.target>17</maven.compiler.target><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding></pro…

Spring 整合MyBatis

创建工程 pom.xml <?xml version"1.0" encoding"UTF-8"?> 4.0.0 <groupId>com.by</groupId> <artifactId>Spring_MyBatis</artifactId> <version>1.0-SNAPSHOT</version><properties><!-- 项目源码…

高可用分布式部署Spark、完整详细部署教程

前言 Spark 是 UC Berkeley AMP Lab 开源的通用分布式并行计算框架。 Spark基于map reduce算法实现的分布式计算&#xff0c;拥有Hadoop MapReduce所具有的优点&#xff1b;但不同于MapReduce的是Job中间输出和结果可以保存在内存中&#xff0c;从而不再需要读写HDFS&#xff…

数据采集有哪些方法?HTTP代理起到什么作用?

在这个数字化的时代&#xff0c;数据就如同生活中不可或缺的元素&#xff0c;我们的行为、喜好、甚至是想法都被转化成了数字化的信息。那么&#xff0c;现代社会是如何进行数据的采集的呢&#xff1f;让我们一同来看看&#xff01; 1. 网络浏览行为的追踪 在我们浏览互联网的…

【Windows】之微软输入法配置小鹤双拼

前言 Windows 自带的输入法微软输入法本身就是个最简洁、最方便的输入法&#xff0c;不需要去安装多余的第三方输入法软件。同时&#xff0c;微软中文拼音输入法支持双拼输入法&#xff0c;但微软自带的双拼输入法不包含小鹤双拼方案的。所以&#xff0c;在这里将会讲解如何配置…

原生微信小程序如何动态修改svg图片颜色及尺寸、宽高(封装svgIcon组件)解决ios不显示问题

最终效果 前言 动态设置Svg图片颜色就是修改Svg源码的path中的fill属性&#xff0c; 通过wx.getFileSystemManager().readFile读取.xlsx文件 ios不显示需要把encoding设置 binary 把文件转成base64 封装svg-icon组件 1、在项目的components下新建svg-icon文件夹&#xff0c;新…

antd Table 动态数据 合并单元格(合并行)

antd Table 组件动态合并单元格 最近处理table的时候 遇到了要合并同一列的几行的情况&#xff0c;比如第一列的前面三行都是同一个对象的名字&#xff0c;此时合并显示比较妥当&#xff0c;但是数据是后端接口来的&#xff0c;而且可以筛选条件&#xff0c;搜索出来的数据就是…

目标检测 | YOLOv5 训练自标注数据集实现迁移学习

Hi&#xff0c;大家好&#xff0c;我是源于花海。本文主要了解 YOLOv5 训练自标注数据集&#xff08;自行车和摩托车两种图像&#xff09;进行目标检测&#xff0c;实现迁移学习。YOLOv5 是一个非常流行的图像识别框架&#xff0c;这里介绍一下使用 YOLOv5 给使用 Labelme 标注…

AI模型部署落地综述(ONNX/NCNN/TensorRT等)

导读 费尽心血训练好的深度学习模型如何给别人展示&#xff1f;只在服务器上运行demo怎么吸引别人的目光&#xff1f;怎么才能让自己的成果落地&#xff1f;这篇文章带你进入模型部署的大门。 0 前言 模型部署的步骤&#xff1a; 训练一个深度学习模型&#xff1b; 使用不同…

NNDL总结

第四章 前馈神经网络 4.1 神经元 人工神经元&#xff0c;简称神经元&#xff0c;是构成神经网络的基本单元。 当>0时&#xff0c;为1&#xff0c;兴奋&#xff1b; 当<0时&#xff0c;为0&#xff0c;抑制。 激活函数的性质 1、连续可导的非线性函数。 2、激活函数及其导…

C语言 B树的分析与实现

本文主要说明了B树的概念、应用以及如何用C语言实现B树。 概述 有使用过数据库的朋友都知道&#xff0c;数据库需要存储大量的数据&#xff0c;并且查询数据的性能也需要一定的保证。那么数据库的底层数据结构是如何实现的呢&#xff0c;就是我们要讨论的B树和B树&#xff0c…
最新文章