背景: 最近接入短剧内容,需要添加短剧合作方。在详情页需要支持添加组件 方案一:标签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-inputv-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,本期结束,关注作者不迷路~持续更新好用的组件~