前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >APICloud AVM框架 封装SKU(商品规格选择)组件

APICloud AVM框架 封装SKU(商品规格选择)组件

作者头像
白鱼赤乌
发布2022-09-02 16:06:02
1.3K0
发布2022-09-02 16:06:02
举报
文章被收录于专栏:案例分享

AVM(Application-View-Model)前端组件化开发模式基于标准Web Components组件化思想,提供包含虚拟DOM和Runtime的编程框架avm.js以及多端统一编译工具,完全兼容Web Components标准,同时兼容Vue和React语法糖编写代码,编译工具将Vue和React相关语法糖编译转换为avm.js代码。

基于标准 Web Components 组件化思想,兼容 Vue / React 语法特性,通过一次编码,分别编译为 App、小程序代码,实现多端开发。

组件功能介绍

该SKU组件可实现商品图片与颜色属性进行联动,通过颜色、版本来控制价格,总价通过购买数量与所选商品价格进行自动计算;可进行缺货设定。

​编辑

上述功能点是通过商品数据结构和代码逻辑进行配合来实现的。

商品数据结构如下:

代码语言:javascript
复制
goodsList:[
				{id:'100016015112',
				image:'https://m.360buyimg.com/mobilecms/s750x750_jfs/t1/210630/17/8651/208682/618a5bd6Eddc8ea0e/b5e55e1a03bc0126.jpg!q80.dpg.web',
				color:'亮黑色',
				status:true,
				guige:[
					{label:'8G+128G',price:'3999',status:false},
					{label:'8G+256G',price:'5999',status:true},
					{label:'16G+512G',price:'6999',status:true},
					{label:'16G+1024G',price:'9999',status:false}
				]},
				{id:'100016015113',
				image:'https://img14.360buyimg.com/n4/jfs/t1/216079/14/3895/201095/618a5c0cEe0b9e2ba/cf5b98fb6128a09e.jpg',
				color:'釉白色',
				status:true,
				guige:[
					{label:'8G+128G',price:'3799',status:true},
					{label:'8G+256G',price:'5799',status:true},
					{label:'16G+512G',price:'6799',status:true},
					{label:'16G+1024G',price:'9799',status:false}
				]},
				{id:'100016015132',
				image:'https://img14.360buyimg.com/n4/jfs/t1/215845/12/3788/221990/618a5c4dEc71cb4c7/7bd6eb8d17830991.jpg',
				color:'秘银色',
				status:true,
                guige:[
					{label:'8G+128G',price:'3599',status:true},
					{label:'8G+256G',price:'5599',status:true},
					{label:'16G+512G',price:'6599',status:true},
					{label:'16G+1024G',price:'9599',status:false}
				]},
				{id:'200016015117',
				image:'https://img14.360buyimg.com/n4/jfs/t1/203247/8/14659/237368/618a5c87Ecc968774/b0bb25331e5e2d1a.jpg',
				color:'夏日胡杨',
                status:false,
				guige:[
					{label:'8G+128G',price:'3899',status:false},
					{label:'8G+256G',price:'5899',status:false},
					{label:'16G+512G',price:'6899',status:false},
					{label:'16G+1024G',price:'9899',status:false}
				]},
				{id:'100013415456',
				image:'https://img14.360buyimg.com/n4/jfs/t1/160950/40/25098/234168/618a5cb9E65ba975e/7f8f93ea7767a51b.jpg',
				color:'冬日暖阳',
				status:true,
                guige:[
					{label:'8G+128G',price:'3199',status:true},
					{label:'8G+256G',price:'5199',status:true},
					{label:'16G+512G',price:'6199',status:true},
					{label:'16G+1024G',price:'9199',status:false}
				]}
			]

每个商品的版本有多条数据,每条数据都对应不同的价格,同时会有字段标识是否有货。

每一个商品也会有字段标识是否有货。

当切换商品属性时,需要通过函数进行判断,以保证在商品缺货的情况下,不能被选中。

代码语言:javascript
复制
setGoods(e){
	// console.log(JSON.stringify(e));
	let item = e.currentTarget.dataset.item;
	if(item.status){
		this.data.selectGoods = item;
		this.data.guigeList = item.guige;
		for (let index = 0; index <  item.guige.length; index++) {
			const element = item.guige[index];
             //保证默认选中的商品是有货的
			if(element.status){
				this.data.selectGuige = element;
				break;
			}
		}
				
	}
},

示例展示

编辑

组件开发

组件文件

easy-sku.stml

代码语言:javascript
复制
<template>
	<view class="easy-sku_container">
		<view class="easy-sku_box">
			<view class="easy-sku_base">
				<view class="easy-sku_base-pic-box">
					<image class="easy-sku_base-pic" src={selectGoods.image} mode="widthFix"></image>
				</view>
				<view class="easy-sku_base-info">
					<text class="easy-sku_base-info-price">¥{selectGuige.price?selectGuige.price*count:0}</text>
					<text class="easy-sku_base-info-num">商品编号:{selectGoods.id}</text>
				</view>
				<view class="easy-sku_close-box" @click="cancel">
					<image class="easy-sku_close-ico" src={ico_close} mode="widthFix"></image>
				</view>
			</view>		
			<scroll-view class="easy-sku_scroll-box" scroll-y>
				<view class="easy-sku_guige-box">
					<view class="easy-sku_guige-title">
						<text class="easy-sku_guige-title-label">颜色</text>
					</view>
					<view class="easy-sku_guige-item-box">
						<view :class="selectGoods.id==item.id?'easy-sku_guige-item--select':'easy-sku_guige-item--default'" v-for="(item,index) in goodsList" data-item={item} @click="setGoods">
							<text :class="selectGoods.id==item.id?'easy-sku_guige-item-label-select':item.status?'easy-sku_guige-item-label--default':'easy-sku_guige-item-label-disable'">{item.color}</text>
						</view>			
					</view>
				</view>
				<view class="easy-sku_guige-box">
					<view class="easy-sku_guige-title">
						<text class="easy-sku_guige-title-label">版本</text>
					</view>
					<view class="easy-sku_guige-item-box">
						<view :class="selectGuige.label==item.label?'easy-sku_guige-item--select':'easy-sku_guige-item--default'" v-for="(item,index) in guigeList" data-item={item} @click="setGuige">
							<text :class="selectGuige.label==item.label?'easy-sku_guige-item-label-select':item.status?'easy-sku_guige-item-label--default':'easy-sku_guige-item-label-disable'">{item.label}</text>
						</view>		
					</view>
				</view>
				<view class="easy-sku_count-box">
					<text class="easy-sku_count-label">购买数量</text>
					<view class="easy-sku_count-result-tool">
						<text :class="count>0?'easy-sku_count-result-item':'easy-sku_count-result-item--disable'" @click="countDel">-</text>
						<input class="easy-sku_count-result-input" placeholder="" keyboard-type="number" v-model="count"/>
						<text class="easy-sku_count-result-item" @click="countAdd">+</text>
					</view>
				</view>
			</scroll-view>
			<view class="easy-sku_bottom">
				<button class="easy-sku_btn" @click="submit">确认</button>
			</view>
			<safe-area></safe-area>	
		</view>		
	</view>
</template>
<script>
export default {
	name: 'easy-sku',
	installed(){
		this.data.selectGoods=this.props.goodsList[0];
		// this.data.selectGuige=this.props.goodsList[0].guige[1];
		this.data.guigeList = this.props.goodsList[0].guige;
	},
	props:{
		goodsList:Array,
	},
	data() {
		return{			
			ico_close:'',
			selectGoods:{},
			selectGuige:{},
			count:1,
			guigeList:[]
		}
	},
	methods: {
		setGoods(e){
			// console.log(JSON.stringify(e));
			let item = e.currentTarget.dataset.item;
			if(item.status){
				this.data.selectGoods = item;
				this.data.guigeList = item.guige;
				for (let index = 0; index <  item.guige.length; index++) {
					const element = item.guige[index];
					if(element.status){
						this.data.selectGuige = element;
						break;
					}
				}
				
			}
		},
		setGuige(e){
			// console.log(JSON.stringify(e));
			let item = e.currentTarget.dataset.item;
			if(item.status){
				this.data.selectGuige = item;
			}
		},
		countAdd(){
			this.data.count++;
		},
		countDel(){
			if(this.data.count>0){
				this.data.count--;
			}	
		},
		cancel(){
			this.fire('cancel','');
		},
		submit(){
			this.fire('submit',{goods:this.data.selectGoods,guige:this.data.selectGuige,count:this.data.count});
		}
	}
}
</script>
<style>
	.easy-sku_container {
		position: absolute;
		height: 100%;
		width: 100%;
		background-color: rgba(0,0,0,0.1);
	}
	.easy-sku_box{
		align-items: center;
		justify-content: space-between;
		position: absolute;
		bottom: 0;
		width: 100%;
		height: 70%;
		background-color: #ffffff;
		border-top-left-radius: 30px;
		border-top-right-radius: 30px;
		padding: 15px;
	}
	.easy-sku_base{
		width: 100%;
		padding: 15px;
	}
	.easy-sku_base-pic{
		width: 80px;
		height: 80px;
	}
	.easy-sku_base{
		flex-flow: row nowrap;
		justify-content: space-between;
		align-items: flex-end;
	}
	.easy-sku_base-info{
		flex: 1;
		padding-left: 10px;
	}
	.easy-sku_base-info-price{
		color: #fa2c19;
		font-size: 16px;
	}
	.easy-sku_base-info-num{
		color: #666666;
		font-size: 12px;
	}
	.easy-sku_close-box{
		height: 100%;
	}
	.easy-sku_close-ico{
		width: 20px;
	}
	.easy-sku_scroll-box{
		width: 100%;
		flex: 1;
	}
	.easy-sku_guige-box{
		width: 100%;		
	}
	.easy-sku_guige-title{
		padding: 10px 0;
	}
	.easy-sku_guige-title-label{
		font-size: 16px;
		font-weight: bolder;
	}
	.easy-sku_guige-item-box{
		flex-flow: row wrap;
		justify-content: flex-start;
		align-items: center;
	}
	.easy-sku_guige-item--default{
		background-color: #f2f2f2;
		border: 0.5px solid #f2f2f2;
		border-radius: 18px;
		padding: 8px 18px;
		margin-right: 10px;
		margin-bottom: 10px;
	}
	.easy-sku_guige-item-label--default{
		font-size: 14px;
		color: #333333;
	}
	.easy-sku_guige-item--select{
		background-color: #fee0dd;
		border: 0.5px solid #fa2c19;
		border-radius: 18px;
		padding: 8px 18px;
		margin-right: 10px;
		margin-bottom: 10px;
	}
	.easy-sku_guige-item-label-select{
		font-size: 14px;
		color: #fa2c19;
	}
	.easy-sku_guige-item-label-disable{
		font-size: 14px;
		color: #b5b5b5;
		/* text-decoration: line-through; */
	}
	.easy-sku_count-box{
		width: 100%;
		flex-flow: row nowrap;
		justify-content: space-between;
		align-items: center;
		margin-top: 10px;
	}
	.easy-sku_count-result-tool{
		flex-flow: row nowrap;
		justify-content: space-between;
		align-items: center;
	}
	.easy-sku_count-result-item{
		font-size: 25px;
		padding: 0 5px;
		color: #333333;
	}
	.easy-sku_count-result-item--disable{
		font-size: 25px;
		padding: 0 5px;
		color: #b5b5b5;
	}
	.easy-sku_count-result-input{
		background-color: #f0f0f0;
		border: 0;
		height: 18px;
		width: 40px;
		text-align: center;
		border-radius: 5px;
	}
	.easy-sku_count-label{
		font-size: 16px;
		color: #333333;
	}
	.easy-sku_bottom{
		width: 100%;
	}
	.easy-sku_btn{
		background-color: #fa2c19;
		border-radius: 21px;
		font-size: 15px;
		color: #fff;
		padding: 10px;
	}
</style>

组件使用说明

本组件是基于AVM.js开发的多端组件,通常同时适配Android、iOS、小程序、H5 , 具体支持情况还要看每个组件的说明文档。

首先需要登录开发平台,http://www.apicloud.com。 通过控制平台右上方的模块Store进入,然后选择AVM组件。

​编辑

​编辑

找到对应模块进入,也可通过搜索栏,通过组件名称关键字进行检索。

​编辑

进入模块详情,点击立即下载下载完整的组件安装包。

​编辑

组件压缩包的文件目录如下

​编辑

也可通过查看模块文档 来了解模块的具体参数,引用的原生模块,注意事项等。

​编辑 具体在项目中的使用步骤是,第一步将压缩文件中的easy-sku.stml文件拷贝到项目的components目录,通过阅读readme.md 文档和查看demo示例文件 demo-easy-sku.stml在需要开发的stml文件中,引入组件文件,完成页面的开发。

demo-easy-sku.stml

代码语言:javascript
复制
<template>
	<view class="page">
		<safe-area></safe-area>
		<view class="item" @click="openSKU">
			<text>选择商品</text>
		</view>                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                      
		<easy-sku
			oncancel="closeSKU"
			onsubmit="getSKU"
			:goodsList="goodsList"
			v-if="isShow"
		>
		</easy-sku>
	</view>
</template>
<script>
import '../../components/easy-sku.stml'
export default {
	name: 'demo-easy-sku',
	apiready(){//like created

	},
	data() {
		return{
			isShow:false,
			goodsList:[
				{id:'100016015112',
				image:'https://m.360buyimg.com/mobilecms/s750x750_jfs/t1/210630/17/8651/208682/618a5bd6Eddc8ea0e/b5e55e1a03bc0126.jpg!q80.dpg.web',
				color:'亮黑色',
				status:true,
				guige:[
					{label:'8G+128G',price:'3999',status:false},
					{label:'8G+256G',price:'5999',status:true},
					{label:'16G+512G',price:'6999',status:true},
					{label:'16G+1024G',price:'9999',status:false}
				]},
				{id:'100016015113',
				image:'https://img14.360buyimg.com/n4/jfs/t1/216079/14/3895/201095/618a5c0cEe0b9e2ba/cf5b98fb6128a09e.jpg',
				color:'釉白色',
				status:true,
				guige:[
					{label:'8G+128G',price:'3799',status:true},
					{label:'8G+256G',price:'5799',status:true},
					{label:'16G+512G',price:'6799',status:true},
					{label:'16G+1024G',price:'9799',status:false}
				]},
				{id:'100016015132',
				image:'https://img14.360buyimg.com/n4/jfs/t1/215845/12/3788/221990/618a5c4dEc71cb4c7/7bd6eb8d17830991.jpg',
				color:'秘银色',
				status:true,guige:[
					{label:'8G+128G',price:'3599',status:true},
					{label:'8G+256G',price:'5599',status:true},
					{label:'16G+512G',price:'6599',status:true},
					{label:'16G+1024G',price:'9599',status:false}
				]},
				{id:'200016015117',
				image:'https://img14.360buyimg.com/n4/jfs/t1/203247/8/14659/237368/618a5c87Ecc968774/b0bb25331e5e2d1a.jpg',
				color:'夏日胡杨',status:false,
				guige:[
					{label:'8G+128G',price:'3899',status:false},
					{label:'8G+256G',price:'5899',status:false},
					{label:'16G+512G',price:'6899',status:false},
					{label:'16G+1024G',price:'9899',status:false}
				]},
				{id:'100013415456',
				image:'https://img14.360buyimg.com/n4/jfs/t1/160950/40/25098/234168/618a5cb9E65ba975e/7f8f93ea7767a51b.jpg',
				color:'冬日暖阳',
				status:true,guige:[
					{label:'8G+128G',price:'3199',status:true},
					{label:'8G+256G',price:'5199',status:true},
					{label:'16G+512G',price:'6199',status:true},
					{label:'16G+1024G',price:'9199',status:false}
				]}
			],
		}
	},
	methods: {
		openSKU(){
			this.data.isShow=true;
		},
		closeSKU(){
			this.data.isShow=false;
		},
		getSKU(e){
			console.log(JSON.stringify(e));
			api.toast({
				msg:'颜色:'+e.detail.goods.color+'/规格:'+e.detail.guige.label+'/数量:'+e.detail.count+'/总价:'+e.detail.count*e.detail.guige.price,
				location:'middle'
			})
			this.data.isShow = false;
		}
	}
}
</script>
<style>
	.page {
		height: 100%;
		background-color: #f6f6f6;
	}
	.item{
		background-color: #ffffff;
		margin: 15px;
		padding: 15px;
		border-radius: 5px;
	}
</style>

如果在AVM组件库中,没有找到实际项目中需要的组件,可以自己尝试封装组件。

这是组件化开发的在线文档地址

​编辑

本文系转载,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文系转载前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 组件功能介绍
  • 示例展示
  • 组件开发
    • 组件文件
      • 组件使用说明
      相关产品与服务
      云开发 CloudBase
      云开发(Tencent CloudBase,TCB)是腾讯云提供的云原生一体化开发环境和工具平台,为200万+企业和开发者提供高可用、自动弹性扩缩的后端云服务,可用于云端一体化开发多种端应用(小程序、公众号、Web 应用等),避免了应用开发过程中繁琐的服务器搭建及运维,开发者可以专注于业务逻辑的实现,开发门槛更低,效率更高。
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档