1.sku多规格效果图

商品的sku涉及到很多动态表单,因为分为单规格和多规格。而每次一到写动态表单就让人头疼。所以特此记录一下,方便自己下次再用也帮助他人减少麻烦。闲话少叙,先看效果图。

 上面的部分输入表单可以无视了,主要还是动态商品规格sku的添加。上代码:

2.在vue框架中使用

<template>
	<div class="goodsdetails">
		<div class="title" style="display: flex;justify-content: space-between;align-items: center;">
			<el-page-header @back="goBack" :content="title"></el-page-header>
			<el-button type="primary" size="small" @click="submit">保存</el-button>
		</div><a-divider />
		<el-descriptions title="基础信息"></el-descriptions>
		<el-form :model="ruleForm" :inline="false"   ref="ruleForm" label-width="100px" class="demo-ruleForm">
		  <el-form-item label="封面图:">
		     <Uploadimg :limit="1"  :key="timer"  :filelist="ruleForm.image"  @getfile="filelist"></Uploadimg>
		  </el-form-item>
		  <el-form-item label="轮播图:">
		     <Uploadimg :limit="3"  :key="timer" :filelist="ruleForm.sliderImage"  @getfile="filelistarr"></Uploadimg>
		  </el-form-item>
		  <el-form-item label="商品分类:" >
		    <el-cascader ref="myCascader"  style="width:360px" @change="getpositionName" v-model="ruleForm.cateId" :options="options" placeholder="请选择分类" :props="{ emitPath:false,checkStrictly: true,value:'id',label:'name'}"  class="ml-4"></el-cascader>
		  </el-form-item>
		  <el-form-item label="商品名称:" >
		    <el-input v-model="ruleForm.storeName"></el-input>
		  </el-form-item>
		  <el-form-item label="商品简介:" >
		    <el-input v-model="ruleForm.storeInfo"></el-input>
		  </el-form-item>
		  <el-form-item label="详细描述:" >
		    <el-input v-model="ruleForm.description"></el-input>
		  </el-form-item>
		  <el-form-item label="返现积分:" >
		    <el-input v-model="ruleForm.giveIntegral" type="number" min=0></el-input>
		  </el-form-item>
		  <el-form-item label="销量:" >
		    <el-input v-model="ruleForm.ficti" type="number" min=0></el-input>
		  </el-form-item>
		  <el-form-item label="关键词:" >
		    <el-tag :key="tag" v-for="tag in dynamicTags" closable :disable-transitions="false" @close="handleClosekey(tag)">
		      {{tag}}
		    </el-tag>
			<el-input
			  class="input-new-tag"
			  v-if="inputVisible"
			  v-model="inputValue"
			  ref="saveTagInput"
			  size="small"
			  @keyup.enter.native="handleInputConfirmkey"
			  @blur="handleInputConfirmkey"
			>
			</el-input>
			<el-button v-else class="button-new-tag" size="small" @click="showInputkey">+ New Tag</el-button>
		  </el-form-item>
		  <el-form-item label="排序:" >
		    <el-input v-model="ruleForm.sort" placeholder="数值越大越靠前" type="number" min=0></el-input>
		  </el-form-item>
		  <el-form-item label="是否多规格:" >
		    <el-switch v-model="ruleForm.specType"></el-switch>
		  </el-form-item>
		  <el-form-item label="是否积分商品:" >
		    <el-switch v-model="ruleForm.integralType"></el-switch>
		  </el-form-item>
		  <!-- 规格 -->
		  <el-divider></el-divider>
		  <div>
		          <div class="goods-spec" v-if="ruleForm.specType">
		            <h3>商品规格</h3>
		            <el-link type="primary" @click="addPrivateSpec" class="goods-spec-add">添加规格</el-link>
		          </div>
		          <div v-if="ruleForm.specType" class="goods-container" v-for="(attr, index) in items" :key="index">
		            <div class="goods-content">
		              <div class="goods-content-box">
		                <div class="goods-content-left">
		                  <el-form label-width="80px" style="width:400px">
		                    <el-form-item label="规格名">
		                      <el-input v-model="attr.value" placeholder="请输入规格名"></el-input>
		                    </el-form-item>
		                    <el-form-item label="规格值">
		                      <el-tag v-for="tag in attr.detail" :key="tag" closable :disable-transitions="false" @close="handleClose(tag, attr)">
		                        {{ tag }}
		                      </el-tag>
		                      <el-input
		                        class="input-new-tag"
		                        v-if="attr.inputVisible"
		                        v-model="attr.inputValue"
		                        :ref="`saveTagInput${index}`"
		                        size="small"
		                        @keyup.enter.native="handleInputConfirm(attr.inputValue, attr)"
		                        @blur="handleInputConfirm(attr.inputValue, attr)"
		                      >
		                      </el-input>
		                      <el-button v-else class="button-new-tag" size="small" @click="showInput(attr, index)">+ 添加</el-button>
		                    </el-form-item>
		                  </el-form>
		                </div>
		                <div class="goods-content-right">
		                  <el-link type="danger" @click="delPrivateSpec(index)">删除规格</el-link>
		                </div>
		              </div>
		            </div>
		          </div>
		  
		          <p style="margin:24px 0 10px 0">价格 / 库存</p>
		          <el-table ref="multipleTable" :data="tableColumnList.tableBodyList" stripe tooltip-effect="dark" style="width: 100%;margin-top:1%;">
		            <el-table-column :label="item.propName" :property="item.prop" v-for="item in tableColumnList.tableHeaderList" :key="item.prop" align="center">
		              <template slot-scope="scope">
		                <span>{{ scope.row[scope.column.property] }}</span>
		              </template>
		            </el-table-column>
					<el-table-column label="商品条码">
					  <template slot-scope="scope">
					    <el-input v-model="scope.row.barCode"></el-input>
					  </template>
					</el-table-column>
					<el-table-column label="价格(元)">
					  <template slot-scope="scope">
					    <el-input v-model.number="scope.row.price"></el-input>
					  </template>
					</el-table-column>
					<el-table-column label="成本价(元)">
					  <template slot-scope="scope">
					    <el-input v-model.number="scope.row.cost"></el-input>
					  </template>
					</el-table-column>
					<el-table-column label="原价(元)">
					  <template slot-scope="scope">
					    <el-input v-model.number="scope.row.otPrice"></el-input>
					  </template>
					</el-table-column>
		            <el-table-column label="库存">
		              <template slot-scope="scope">
		                <el-input v-model.number="scope.row.stock"></el-input>
		              </template>
		            </el-table-column>
		            <el-table-column label="体积">
		              <template slot-scope="scope">
		                <el-input v-model="scope.row.volume"></el-input>
		              </template>
		            </el-table-column>
					<el-table-column label="重量">
					  <template slot-scope="scope">
					    <el-input v-model="scope.row.weight"></el-input>
					  </template>
					</el-table-column>
					<el-table-column label="兑换所需积分" v-if="ruleForm.integralType">
					  <template slot-scope="scope">
					    <el-input v-model="scope.row.integral"></el-input>
					  </template>
					</el-table-column>
		          </el-table>
		  </div>
		  <!-- <el-button type="primary" style="width: 20%;margin-left: 30%;margin-top: 20px;">保存</el-button> -->
		</el-form>
	</div>
</template>

<script>
	import Uploadimg from "@/components/Uploadimg";
	export default {
	    data(){
			return{
				ruleForm: {
					cateId:'',
					cateName:'',
					image:[],
					sliderImage:[],
				    storeName: '',
				    storeInfo: '',
				    description: '',
				    giveIntegral: '',
				    ficti: '',
				    sort: '',
				    specType: false,
					integralType:false,
				},
				dynamicTags: ['标签一', '标签二'],  //关键词
				inputVisible: false,
				inputValue: '',
				tableColumnList: {
				            tableHeaderList: [],
				            tableBodyList: [
								{
									
								}
							]
				            // inventory: ''
				          },
				items: [    //sku属性
				        {
				            value: '', //规格名
				            detail: [], //规格值数组
				            inputVisible: false,
				            inputValue: ''
				        }
				],
				options:[],   //商品类列表
				id:"",   //无则添加有则编辑
				timer:'',  //刷新子组件
				title:"",
			}
		},
		mounted(){
			this.title="添加商品"
			if(this.$route.query.id){
				this.id=this.$route.query.id;
				this.title="商品信息"
				this.getdetails();
			}
			this.getoptions();
			
		},
		computed: {
		          // 计算规格
		          calculateAttribute() {
		            // 初始化
		            let obj = {}
		            this.items.forEach((item) => {
		              // 判断有没有输入规格名
		              if (item.value) {
		                //规格名:规格值     //'颜色':'尺寸'
		                obj[item.value] = item.detail
		              }
		            })
		            return obj
		          }
		},
		watch: {
		          // 监听规格数据
		          calculateAttribute(newVal) {
					if(!this.ruleForm.specType){return;}
					this.attribute(newVal);
		//             let cloneNewVal = JSON.parse(JSON.stringify(newVal))
		//             let attrName = [] //规格名数组
		//             let attrValue = [] //规格值数组
		//             for (let key in cloneNewVal) {
		//               attrName.push(key)
		//               attrValue.push(cloneNewVal[key])
		//             }
					
		//             // 表格内容数据(笛卡尔积算法)
		//             let finalArr = this.cartesianProductOf(...attrValue)
		
		//             let tableObj = {
		//               tableBodyList: [],
		//               tableHeaderList: []
		//             }
		//             // 表格内容
		//             tableObj.tableBodyList = finalArr.map((item) => {
		//               let obj = {
		//                barCode: '',   //商品条码
		//                cost:0,        //成本价
		//                otPrice:0,     //原价
		//                price:0,       //现价
		//                stock: 0,	   //库存
		//                volume:'',     //体积
		//                weight:'',     //重量
		//               }
		//               for (let i = 0; i < item.length; i++) {
		//                 obj[attrName[i]] = item[i]
		//               }
		//               return obj
		//             })
		//             this.tableColumnList.tableBodyList = tableObj.tableBodyList //表格内容数据
		
		//             // 表头
		//             let skuTableArr = Object.keys(newVal)
		//             tableObj.tableHeaderList = skuTableArr.map((item) => {
		//               return {
		//                 prop: item,
		//                 propName: item
		//               }
		//             })
		//             this.tableColumnList.tableHeaderList = tableObj.tableHeaderList // 表头
		          },
				  'ruleForm.specType':{
					  deep:true,
					  handler:function(newV,oldV){
					        if(!newV){
								this.tableColumnList= this.$options.data().tableColumnList;
								this.items=this.$options.data().items;
							}
					  }
				  }
		},
		methods: {
			getpositionName(){
				 const info = this.$refs.myCascader.getCheckedNodes()[0];
				 this.ruleForm.cateName=info.label;
			},
			getoptions(){	//获取商品分类列表
				const that = this;
				this.$api.categorylist({}).then(function (response) {
						if(response.code==0){
							that.options=that.removeEmptyChildren(response.data);
						}else{
							that.$message({
								 message: response.message,
								 center: true,
								 type: 'warning',
							});
						}
				})
			},
			removeEmptyChildren (node) {   //杀生
				var that=this;
				node.forEach(item => {
				if ('children' in item && item.children.length === 0) {
					delete item.children
				} else if ('children' in item && item.children.length) {
					that.removeEmptyChildren(item.children)
				}
				})
				return node;
			},
			attribute(newVal){
				var that=this;
				this.$api.goodssku(that.items).then(res=>{
					if(res.code==0){
						// console.log("属性",res);
						let cloneNewVal = JSON.parse(JSON.stringify(newVal))
						let attrName = [] //规格名数组
						let attrValue = [] //规格值数组
						for (let key in cloneNewVal) {
						  attrName.push(key)
						  attrValue.push(cloneNewVal[key])
						}
						
						// 表格内容数据(笛卡尔积算法)
						let finalArr = this.cartesianProductOf(...attrValue)	
						let tableObj = {
						  tableBodyList: [],
						  tableHeaderList: []
						}
						// 表格内容
						tableObj.tableBodyList = res.data.map((item) => {
						  var detail=item.detail;
						  return Object.assign(item, detail);
						})
						this.tableColumnList.tableBodyList = tableObj.tableBodyList //表格内容数据
						this.tableColumnList.tableBodyList = res.data;
						// 表头
						let skuTableArr = Object.keys(newVal)
						tableObj.tableHeaderList = skuTableArr.map((item) => {
						  return {
						    prop: item,
						    propName: item
						  }
						})
						this.tableColumnList.tableHeaderList = tableObj.tableHeaderList // 表头
					}else{
						this.$message({
							 message: res.message,
							 center: true,
						});
					}
				}).catch(err=>{console.log("网络错误",err)})
			},
			//上传的图片列表
			filelist(list){
				console.log("封面图片:",list)
				this.ruleForm.image=list;
			},
			filelistarr(list){
				console.log("轮播图片:",list)
				this.ruleForm.sliderImage=list;
			},
			goBack(){
				 this.$router.go(-1);
			},
			//关键词操作
			handleClosekey(tag) {
			        this.dynamicTags.splice(this.dynamicTags.indexOf(tag), 1);
			},
			
			showInputkey() {
			    this.inputVisible = true;
			    this.$nextTick(_ => {
			          this.$refs.saveTagInput.$refs.input.focus();
			    });
			},
			handleInputConfirmkey() {
			    let inputValue = this.inputValue;
			    if (inputValue) {
			        this.dynamicTags.push(inputValue);
			    }
			    this.inputVisible = false;
			    this.inputValue = '';
			},
			//获取详情
			getdetails(){
				var that=this;
				this.$api.goodsdetails({id:that.id}).then(res=>{
					if(res.code==0){
						console.log("商品:",res);
						that.ruleForm=res.data;
						that.ruleForm.image=res.data.image.split(",");
						that.ruleForm.sliderImage=res.data.sliderImage.split(",");
						that.dynamicTags=res.data.keyword.split(",");
						that.items=res.data.items.map(item=>{
							item.inputValue="";
							item.inputVisible=false;
							return item;
						})
						setTimeout(()=>{
							let tablearr=[];
							for(var i=0; i<res.data.attrs.length;i++){
								tablearr.push(Object.assign(that.tableColumnList.tableBodyList[i],res.data.attrs[i]));
							}
							that.tableColumnList.tableBodyList=tablearr;
						},500)
						
						that.timer = new Date().getTime();
					}else{
						this.$message({
							 message: res.message,
							 center: true,
						});
					}
				}).catch(err=>{console.log("网络错误",err)})
			},
			submit(){
				var that=this;
				var obj;
				if(this.id){  //编辑
					obj=JSON.parse(JSON.stringify(that.ruleForm));
					obj.id=that.id;
					obj.items=that.items;
					obj.attrs=that.tableColumnList.tableBodyList;
					obj.keyword=that.dynamicTags.toString();
					obj.image=obj.image.toString();
					obj.sliderImage=obj.sliderImage.toString();
					this.$api.goodsedit(obj).then(res=>{
						if(res.code==0){
							this.$message({
								 type:"success",
								 message: res.message,
								 center: true,
							});
							that.$router.push({path:'/product/goods'})
						}else{
							this.$message({
								 message: res.message,
								 center: true,
							});
						}
					}).catch(err=>{console.log("网络错误",err)})
				}else{
					obj=JSON.parse(JSON.stringify(that.ruleForm));
					
					obj.items=that.items;
					obj.attrs=that.tableColumnList.tableBodyList;
					obj.keyword=that.dynamicTags.toString();
					obj.image=obj.image.toString();
					obj.sliderImage=obj.sliderImage.toString();
					this.$api.goodsadd(obj).then(res=>{
						if(res.code==0){
							this.$message({
								 type:"success",
								 message: res.message,
								 center: true,
							});
							that.$router.push({path:'/product/goods'})
						}else{
							this.$message({
								 message: res.message,
								 center: true,
							});
						}
					}).catch(err=>{console.log("网络错误",err)})
				}
			},
			/*****规格*****/
		    // 添加规格
		    addPrivateSpec(index) {
		            this.items.push({
		              value: '',
		              detail: [],
		              inputVisible: false,
		              inputValue: ''
		            })
		    },
		    delPrivateSpec(index) {
		            this.items.splice(index, 1)
		    },
		    handleInputConfirm(val, attr) {
		        if (val) {
		            attr.detail.push(val)
		        }
		        attr.inputVisible = false
		        attr.inputValue = ''
		    },
		    handleClose(tag, item) {
		            item.detail.splice(item.detail.indexOf(tag), 1)
		    },
		    showInput(attr, index) {
		        attr.inputVisible = true
		            this.$nextTick((_) => {
		              this.$refs[`saveTagInput${index}`][0].$refs.input.focus()
		        })
		    },
		    // 笛卡尔积算法
		    cartesianProductOf(...args) {
		            return args.reduce(
		              (total, current) => {
		                let ret = []
		                total.forEach((a) => {
		                  current.forEach((b) => {
		                    ret.push(a.concat([b]))
		                  })
		                })
		                return ret
		              },
		              [[]]
		            )
		    }

		},
		components:{
			Uploadimg:Uploadimg,
		}
	};
</script>
<style scoped>
	.goodsdetails{
		height: 100%;
		width: 100%;
		background: #FFFFFF;
		padding: 30px;
	}
	.goodsdetails .demo-ruleForm{
		width: 100%;
		/* display: flex;
		align-items: center;
		justify-content: space-between;
		flex-wrap: wrap; */
	}
	.goodsdetails .demo-ruleForm>div{
		width: 80%;
	}
	.goodsdetails .el-tag + .el-tag {
	    margin-left: 10px;
	  }
	.goodsdetails  .button-new-tag {
	    margin-left: 10px;
	    height: 32px;
	    line-height: 30px;
	    padding-top: 0;
	    padding-bottom: 0;
	  }
	.goodsdetails  .input-new-tag {
	    width: 90px;
	    margin-left: 10px;
	    vertical-align: bottom;
	  }
	/* .goodsdetails .uploadimg{
		display: flex;
		width: 80vw;
	} */
	.goods-spec {
	        display: flex;
	        justify-content: space-between;
	        margin-bottom: 10px;
	 }
	      .goods-spec .goods-spec-add {
	        margin-right: 15px;
	      }
	
	      .goods-container .button-new-tag {
	        height: 32px;
	        line-height: 30px;
	        padding-top: 0;
	        padding-bottom: 0;
	      }
	      .goods-container .input-new-tag {
	        width: 90px;
	        margin-right: 10px;
	        vertical-align: bottom;
	      }
	      .goods-container .el-tag {
	        margin-right: 10px;
	      }
	      .goods-container .goods-content {
	        margin-bottom: 10px;
	        padding: 14px;
	        border: 1px solid #ebeef5;
	        border-radius: 4px;
	        background-color: #fcfcfc;
	      }
	      .goods-content .goods-content-box {
	        display: flex;
	        align-items: center;
	      }
	      .goods-content-box .goods-content-left {
	        flex: 1;
	      }
</style>

这是单个的vue文件,在使用之前,先确保你项目是否使用了element Ui组件。使用时直接复制到页面就行。可能会报错。

报错1:这里我使用了Uploadimg图片上传组件。主要是上传图片到阿里云oss。删掉即可。

报错2:页面加载时请求了接口,调用了方法this.getoptions();主要是拉取商品分类列表数据。注释和删掉即可。

其他的就是逻辑代码了。可以happy~奔放一下看到具体效果。

码字不易,转载请注明出处,谢谢~

Logo

前往低代码交流专区

更多推荐