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

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

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

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

 

组件功能介绍

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

image.gif编辑

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

商品数据结构如下:

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}
        ]}
      ]

image.gif

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

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

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

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;
      }
    }
  }
},

image.gif

示例展示

image.gif编辑

组件开发

组件文件

easy-sku.stml

<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:'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABQAAAAUCAYAAACNiR0NAAAAjUlEQVR42mNgGNlg1qxZbTNnzrxFgvqnQJyLUwHQsKtA/B+InxEyDKjmI0gt0MD9hBQ+ImQozDAgvkyUV/AZSrJh+Awl2zBshlJsGBZDKTcMmjR+Ihn4kVqGHQS5jiJDkQ1D8j55hmIzjGxD8RlGsqFABe8IGYbF0Pv4FAUAXbiIhCS1ac6cOXYjvEwFAL7uy3s7L72GAAAAAElFTkSuQmCC',
      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>

image.gif

组件使用说明

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

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

image.gif编辑

image.gif编辑

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

image.gif编辑

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

image.gif编辑

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

image.gif编辑

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

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

demo-easy-sku.stml

<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>

image.gif

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

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

image.gif编辑


相关文章
|
6月前
|
监控 前端开发 JavaScript
实战篇:商品API接口在跨平台销售中的有效运用与案例解析
随着电子商务的蓬勃发展,企业为了扩大市场覆盖面,经常需要在多个在线平台上展示和销售产品。然而,手工管理多个平台的库存、价格、商品描述等信息既耗时又容易出错。商品API接口在这一背景下显得尤为重要,它能够帮助企业在不同的销售平台之间实现商品信息的高效同步和管理。本文将通过具体的淘宝API接口使用案例,展示如何在跨平台销售中有效利用商品API接口,以及如何通过代码实现数据的统一管理。
|
8天前
|
供应链 数据挖掘 API
电商API接口介绍——sku接口概述
商品SKU(Stock Keeping Unit)接口是电商API接口中的一种,专门用于获取商品的SKU信息。SKU是库存量单位,用于区分同一商品的不同规格、颜色、尺寸等属性。通过商品SKU接口,开发者可以获取商品的SKU列表、SKU属性、库存数量等详细信息。
|
29天前
|
前端开发 JavaScript API
Vue3商品SKU多规格编辑组件
Vue3商品SKU多规格编辑组件
59 5
|
2月前
|
XML 前端开发 Java
谷粒商城笔记+踩坑(5)——商品服务-属性分组、品牌关联分类,spu+sku+分页拦截器
SPU和SKU、属性分组的增删改查、QueryWrapper的and和or用法、获取当前品牌关联的所有分类
谷粒商城笔记+踩坑(5)——商品服务-属性分组、品牌关联分类,spu+sku+分页拦截器
|
3月前
|
数据采集 API 开发工具
淘系商品详情数据解析(属性youhui券sku详情图等)API接口开发系列
在电商领域,特别是像淘宝(淘系)这样的平台,商品详情数据对于商家、开发者以及数据分析师来说至关重要。这些数据包括但不限于商品属性、优惠券信息、SKU(Stock Keeping Unit)详情、商品图片、售后保障等。然而,直接访问淘宝的内部API接口通常需要特定的权限和认证,这通常只对淘宝的合作伙伴或内部开发者开放。 不过,对于需要这些数据的第三方开发者或商家,有几种方式可以间接获取或解析淘系商品详情数据: ——在成长的路上,我们都是同行者。这篇关于商品详情API接口的文章,希望能帮助到您。期待与您继续分享更多API接口的知识,请记得关注Anzexi58哦!
|
3月前
|
数据采集 JSON API
淘系商品详情图属性sku价格解析,API接口系列
淘宝(Taobao)作为阿里巴巴集团旗下的电商平台,其商品详情图、属性、SKU和价格的采集通常不直接通过公开的API接口来实现,因为淘宝的API主要面向商家和开发者提供店铺管理、订单处理、物流查询等功能,并不直接提供商品详情页的完整数据抓取接口
|
4月前
|
XML JSON API
开发者必备:淘宝商品列表接口集成全攻略
淘宝开放平台提供的商品列表数据接口让开发者编程获取商品列表数据。接口支持按关键词、类目等查询条件获取商品详情,包括标题、价格等信息。具备灵活性高、数据丰富及操作便捷等特点。使用流程包括注册账号、构建并发送HTTP请求及处理响应数据。可用于电商数据分析、商品推荐等场景。开发者需遵守规定确保数据安全合法。[体验API](c0b.cc/R4rbK2)
|
4月前
|
供应链 搜索推荐 数据挖掘
微店商品详情数据接口(micro.item_get)丨微店API接口指南
`micro.item_get`接口是微店API的关键工具,让开发者能获取商品详情,包括名称、价格、描述、图片、销量和SKU,用于电商同步、数据分析、个性化营销和提升购物体验。此接口加速了数据驱动的决策和业务优化。
|
6月前
|
JavaScript Java 测试技术
基于SpringBoot+Vue的免税商品优选购物商城的设计与实现(源码+lw+部署文档+讲解等)
基于SpringBoot+Vue的免税商品优选购物商城的设计与实现(源码+lw+部署文档+讲解等)
|
开发者
【 uniapp - 黑马优购 | 商品列表 】如何实现数据获取、结构渲染、自定义组件的封装
【 uniapp - 黑马优购 | 商品列表 】如何实现数据获取、结构渲染、自定义组件的封装
368 0