热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

仿得物微信小程序(动手就会,含源码)

前言最近正在学习微信小程序开发,也是与两个同学一起合作着手仿做得物APP微信小程序。这里主要分享一下我的学习过程及踩过的一些坑,希望对您有所帮助。006

前言

最近正在学习微信小程序开发,也是与两个同学一起合作着手仿做得物APP微信小程序。这里主要分享一下我的学习过程及踩过的一些坑,希望对您有所帮助。

006.gif

开发准备

  • 微信开发者工具

  • VScode代码编辑器

  • 得物APP微信小程序

  • 有赞vant组件库[1]

  • 阿里巴巴矢量图标库[2]

  • markman[3](取色量距)

总体架构

  • 该项目基于小程序云开发,使用的模板是云开发快速启动模板[4]由于是个全栈项目,前端使用小程序所支持的wxml + wxss + js开发模式,命名采用BEM[5]命名规范。后台则是借助云数据库进行数据管理。

项目中我负责的部分主要如下(一些数据为固定数据写在config中,js文件通过module.exports暴露,需要引用时在页面对应js头部引入,例const {} = require('../../../../config/buys'))。项目中我使用的较多vant组件,需要在构建npm包时引入vant,详情可见有赞vant的npm安装[6]。页面使用第三方组件时须在对应json文件中声明,为了不做重复工作可直接在app.json中声明。例:("usingComponents": "van-search": "@vant/weapp/search/index"}

    |-config  对应数据|-assem.js   |-buys.js    |-detail.js  |-kind.js    |-search.js  |-pages|-buy_page|-page|-assem   筛选排序页|-buy     购买首页|-detail  商品详情页|-kinds   品牌分类页|-produce 鉴别简介页|-search  搜索页

项目规划

  • 在做该小程序之前,我先是分析每个页面对应功能,了解这款小程序的交互细节,清楚数据集合数据项。这样大概可以分为分析页面,创建数据集合,解构页面基本布局,数据绑定及跳转四步来展开。

参照得物APP微信小程序,下面是我的小程序的tabBar。(有点糙,但是还能看????)

000.png

  "tabBar": {"selectedColor": "#000000","borderStyle": "white","backgroundColor": "#fff","list": [{"text": "购买","pagePath": "pages/buy_page/page/buy/buy","iconPath": "images/buy.png","selectedIconPath": "images/buy_active.png"},{"text": "鉴别查询","pagePath": "pages/disting/disting","iconPath": "images/disting.png","selectedIconPath": "images/disting_active.png"},{"text": "洗护","pagePath": "pages/wash/wash","iconPath": "images/wash.png","selectedIconPath": "images/wash_active.png"},{"text": "我","pagePath": "pages/my_page/my/my","iconPath": "images/my.png","selectedIconPath": "images/my_active.png"}]},

云数据库

云数据库是一种NoSQL数据库。每一张表是一个集合。对于我的项目部分,我主要建立了一个商品集合。

dewu_goods 商品表 用于存储创商品的信息 - _id - amway  是否为推荐- brand  品牌- buyer  已购买人数- ctime  数据创建时间- digest 详情介绍- img    详情图- pic    商品展示图- kind   种类- price  价格- sex    适应人群- title  简介- type   首页索引

建立数据集合后需修改数据权限才可正常访问。

可对在数据库中进行这些操作,注意导入数据格式需要是.csv.json文件,可先用excel表格建立数据集合如何转化成对应格式文件直接导入数据库。

0101.png

const db = wx.cloud.database() //云数据库
const dewuCollection = db.collection('dewu') //在js文件中导入数据集合

项目解构

以下是我主要实现的得物APP小程序界面

接下来对每个页面的细节进行解构。

购买首页

000.gif

购买首页样式

00.png

    正品保障{{}}


商品项van-grid-item中采用绝对定位。tips中将direction属性设置为horizontal,可以让宫格的内容呈横向排列。搜索框设置disabled属性为禁用状态解决单击自动聚焦的问题。在使用van-grid布局时自定义每一项的属性需设置use-slot属性,否则不生效。

这个页面布局并不复杂,不过我在写这个布局时还是遇到了坑(感觉是自己跳进去的 我太????了)。在做dewu-hd吸顶时我是直接用van-sticky包起来实现,但是实际效果是tabs也需要固定在dewu-hd下面。这里不可以使用同上的方法,实际效果会使得整个van-tabs吸顶导致页面无法滑动。其实在这里只需要给van-tabs添加一个sticky属性并且设置offset-top,注意这两个属性需一起使用才能生效。

获取商品项

  async onLoad() {this.proData()   //获取推荐数据项this.shoeData()  //获取鞋类数据项},proData() {const {data} = await dewuCollection.where({amway: db.command.eq('TRUE')}).field({          //获取指定数据项,提升性能_id:true,pic:true,title:true,buyer:true,price:true})  .get()// console.log(data);this.setData({produces: data,})}shoeData() {let data1 = await dewuCollection.where({type: 1}) .get()// console.log(data1.data);this.setData({shoes: data1.data})}

绑定详情页

  gotoDetail(e) {// console.log(e);wx.navigateTo({url: '/pages/buy_page/page/detail/detail?id='+e.currentTarget.dataset.id,})},

利用商品_id属性唯一,当设定数据项id等于_id时跳转到详情页且展示对应数据。

商品详情页

001.gif

商品详情页样式

03.png


{{img.digest}}¥{{img.price}}选择尺码请选择尺码


整体分为detail_hddetail_bd两部分。自定义swiper需设置dot对应展示图片并更改样式,circular属性设置是否启用滑块切换动画,这里使用三目运算符判断是否添加新的样式类名。在定义商品价格的样式的时候可以通过first-letter伪元素来定义¥符号样式。引用组件van-goods-action使得购买按钮吸底。

05.png

--{{item.price}}请选择商品已选 {{item.size}}{{item.size}}¥{{item.price}}

使用van-popup组件,给对应标签设置事件即可绑定弹出。例:。三目运算符设置默认样式并且控制选中边框样式,设置closeable属性启用关闭按钮。square设置van-grid-item为方形,gutter设置格子间距。

06.png

相关推荐

设置detail_produce-hd吸顶,给右侧关闭icon绑定bind:close="onClose"事件。

获取商品详情

  async onLoad(options) {  //获取对应_id的商品数据console.log(options);let id = options.idconsole.log(id);wx.cloud.database().collection('dewu') .doc(id)             .get().then(res => {console.log(res);this.setData({img :res.data})})},

弹出层

  showPopup() {   //显示弹出层this.setData({ show: true,});},onClose() {     //关闭弹出层this.setData({ show: false,});},

选择尺码

  pickSize(e) {let flag = e.currentTarget.dataset.flaglet index = e.currentTarget.dataset.indexif(flag==index) {   this.setData({activeSizeIndex: -1,flag: -1}) }else {this.setData({activeSizeIndex: index,flag: index}) }},

点击尺码,flag==index即为选中状态,再次点击时或者点击其他尺码时设置为非选中状态,否则使flag等于index,使其变成选中状态。

搜索页

002.gif

搜索页样式

01.png

 0}}">历史搜索{{}}


搜索页面主要分为头部搜索框和内容(搜索推荐,历史记录和搜索到的商品列表)两部分。这里用van-sticky包装搜索框使吸顶,内容部分则用block标签包装,利用wx:if这个控制属性来判断是否显示。

搜索记录

  async onSearch(e) {// console.log(e);if (!e.detail.trim()) {wx.showToast({title: '请输入商品名',})return}let {value, historyList} = this.dataif(historyList.indexOf(value) !== -1) {historyList.splice(historyList.indexOf(value), 1)}historyList.unshift(value)this.setData({historyList})wx.setStorageSync('value', historyList)let keyword = e.detail.trim()let results = await dewuCollection.where({title: db.RegExp({regexp: keyword,options: 'i'})}).get()if (results.data.length == 0 || keyword == '') {wx.showToast({title: '不存在'+keyword,})}else {await dewuCollection.where({title: db.RegExp({regexp: keyword,options: 'i'})}).orderBy('hot', 'desc').get().then(res => {console.log(res);this.setData({results: res.data})})}},
onLoad() {this.getSearchHistory()  //获取历史搜索},
getSearchHistory() {let historyList = wx.getStorageSync('value')if(historyList) {this.setData({historyList})}},

页面加载时从本地storage中获取历史搜索记录,在确定搜索onSearch时判断value是否为空,将合法value插入historyList中,这里使用的时unshift方法,这样可以保证最近的搜索记录展示在前面,利用正则表达式模糊查询数据库中符合的项存入数组results中,当results.length > 0时显示商品列表。利用wx.setStorageSyncvalue存入缓存,wx.getStorageSync获取打印出来。通过indexOf方法判断value是否已经存在,是则删除historyList中的该项。

历史搜索

  async historySearch(e) {// console.log(e);let historyList = this.data.historyListlet value = historyList[e.currentTarget.dataset.index]this.setData({value,               //修改valueshowHotList: false,  //隐藏热门搜索showHistory: false,  //隐藏历史搜索results: []          //清空商品列表})},

点击历史搜索项时setData使对应值改变,再调用onSearch方法。

清空控件

  onClear() {this.setData({results: [],value: '',showHotList: true,showHistory: true});},onChange(e) {  //search框输入改变时实时修改数据// console.log(e.detail);this.setData({value: e.detail,showHotList: false,showHistory: false,results: []})// console.log(this.data.showHotList);if (this.data.value=='') {this.setData({showHotList: true,showHistory: true})}},

清空搜索历史

  deleteSearchHistory() {wx.showModal({content: '确认清空历史记录',success: (res) => {if(res.confirm) {this.setData({historyList: []})}}})wx.removeStorageSync('value')},

点击删除icon弹出对话框wx.showModal实现交互,用户点击确定则清空historyList并利用wx.removeStorageSync将本地存储的历史记录删除。

品牌分类页

003.gif

分类页样式

07.png

{{}}{{}}{{}}


分类页面主要是使用了scroll-view设置竖向滚动,点击左侧scroll-view-left-item时该项变为得物色(#00cbcc)并显示对应的品牌种类项kindsItem。整体采用flex布局,这里的坑是scroll-view-left应该把font-size设为0,在子元素scroll-view-left-item中设置font,避免块元素边距影响布局。

初始化品类

  onLoad: function (options) {this.setData({kindNav: kindNav,kindall: kindItem,// console.log(this.data.kindall);let kinds=[];// console.log(this.data.kindall)this.data.kindall.forEach(kind => { //循环从所有品类中获取对应kindNav的并存入数组中if(kind.camptype == 0) {kinds.push(kind)}})this.setData({kindItem: kinds,})}, )},

选择分类

  changeKinds(e) {console.log(e);let {index, type} = e.currentTarget.dataset;console.log(index, type);//index与推荐品牌的索引有关。type与kind.js的camptype有关this.setData({activeNavIndex: index,})let title=[]this.data.kindTitles.forEach(kindTitle => {if(index == kindTitle.titletype) {title.push(kindTitle)}})this.setData({kindItem: kinds,})},

绑定筛选页

  gotoAssem(e) {// console.log(e);  利用kind属性值唯一(buy页面tabs的title)wx.navigateTo({url: '/pages/buy_page/page/assem/assem?title='+e.currentTarget.dataset.title,})},

筛选排序页

004.gif

排序页样式

08.png

{{item.title}}


tab使用flex布局。goods部分布局参照buy页面的商品布局。

011.png

  {{}}

这里使用van-collapse组件做折叠面板时有个坑,不应该将van-grid内容部分放在van-collapse-item中,应与其同级,否则会在该单元格下形成留白且无法正常显示内容,多次尝试后还是放在外面方便实现效果。

初始商品排序

  async onLoad(options) {// console.log(options);let title = options.titlelet data1 = await dewuCollection.where({kind: title        //绑定跳转时(kind唯一)获取对应数据}) .get()// console.log(data1);this.setData({goods: data1.data,titles: title})},

基本排序

  async changeItem(e) {  // console.log(e);let index = e.currentTarget.dataset.index  //index对应排序方式this.setData({activeTabIndex: index})// console.log(index);if(index == 1) {                        //销量排序await dewuCollection.where({kind: this.data.titles}).orderBy('buyer', 'desc') .get().then(res => {this.setData({goods: res.data,index: index})// console.log(this.data.index);})}if(index == 0) {                        //综合排序await dewuCollection.where({kind: this.data.titles}).get().then(res => {this.setData({goods: res.data})})}if(index == 2 && this.data.flag == -1) {  //价格降序排序await dewuCollection.where({kind: this.data.titles}).orderBy('price', 'desc') .get().then(res => {this.setData({goods: res.data,flag: 1})})return}if(index == 3) {                         //创建时间排序await dewuCollection.where({kind: this.data.titles}).orderBy('ctime', 'desc') .get().then(res => {this.setData({goods: res.data})})}if(index == 4) {                         //弹出筛选层this.setData({ show: true,})}else if(index == 2 && this.data.flag == 1) {    //价格升序排序await dewuCollection.where({kind: this.data.titles}).orderBy('price', 'asc') .get().then(res => {this.setData({goods: res.data,flag: -1})})}},

设置一个flag属性默认值为-1,flag==-1时点击价格降序排序并设置flag==1flag==1时点击价格升序排序并设置flag==-1

筛选排序

  pick(e) {let flag = e.currentTarget.dataset.flaglet index = e.currentTarget.dataset.indexlet cd = this.data.human[index].kindif(flag==index) {this.setData({activeIndex1: -1,flag1: -1,cd1: ''}) }else {this.setData({activeIndex1: index,flag1: index,cd1: cd}) }},

筛选重置

  replace() {    // 点击重置按钮将所有筛选条件回复默认this.setData({flag1: -1,activeIndex1: -1,flag2: -1,activeIndex2: -1,flag3: -1,activeIndex3: -1,cd1: '',cd2: '',cd3: 0,cd4: 10000000,})},

这里有一个坑是,不可在data中声明(num:Infinity),这里无穷大并不会生效,目前优化是声明为常量.

确认筛选

  async ischeck() {   //点击确定按钮进行筛选显示结果let cd3 = Number(this.data.cd3)let cd4 = Number(this.data.cd4)==0?1000000:Number(this.data.cd4)let index = Number(this.data.index)if(this.data.cd1!='' && this.data.cd2!=''){await dewuCollection.where({kind: this.data.titles,sex: this.data.cd1,brand: this.data.cd2,price: _.gt(cd3).and(_.lt(cd4)),}).get().then(res => {this.setData({goods: res.data,show: false,})})return}  },

难点排坑

    gotoDetail(e) {// console.log(e);wx.navigateTo({url: '/pages/buy_page/page/detail/detail?id='+e.currentTarget.dataset.id,})},      

跳转到详情页且保留对应数据项。这里利用_id唯一,将每一项的_id赋给data-id,当id相等时才能跳转并接受对应_id的数据。

        {{item.size}}¥{{item.price}}pickSize(e) {let flag = e.currentTarget.dataset.flaglet index = e.currentTarget.dataset.indexif(flag==index) {this.setData({activeSizeIndex: -1,flag: -1}) }else {this.setData({activeSizeIndex: index,flag: index}) }},

点击尺码时选中并更改text,再次点击该项则重置样式,若点击其他项则取消选中,选中被点击项。这里通过多设一个flag,结合index双重控制是否选中。

      {{item.text}}changeKinds(e) {console.log(e);let {index, type} = e.currentTarget.dataset;console.log(index, type);//index与推荐品牌的索引有关。type与kind.js的camptypethis.setData({activeNavIndex: index,})let kinds = []this.data.kindall.forEach(kind => {if(kind.camptype == type) {kinds.push(kind)}})this.setData({kindItem: kinds,})}

绑定typekind.camptype,当点击项改变时,将当前项index赋给activeNavIndex,用kindall存储所有数据项,使用foreach循环遍历kindall,将满足条件kind.camptype==type的数据存入一个数组中kinds,再将setData即可。

  deleteSearchHistory() {wx.showModal({content: '确认清空历史记录',success: (res) => {if(res.confirm) {this.setData({historyList: []})}}})wx.removeStorageSync('value')},

清空历史记录时不仅将historyList设为空,且利用wx.removeStorageSync将本地存储的缓存清除。

小建议

在自己写项目时,多使用console.log()打印,跟进数据变化;多查看文档w3cschool[7],微信开发文档[8],Vant-Weapp[9]。

源码

本项目源码[10]:https://gitee.com/onepiece1205/dewu_weapp

结语

写项目的过程对我来说是一个挑战,毕竟第一次专注于合作做项目,项目中遇到的bug会烦人但是坚持写功能后是非常有成就感的,非常感谢在我写项目过程中帮助过我的老师和同学。如果你喜欢我的这篇文章或者看到这里对你有些许帮助,不妨点个赞吧????!同时也非常希望看到文章的你能给我一些建议,期待与你一起讨论学习微信小程序!(点击????了解更多[11])

005.gif

关于本文

来源: onepiece1205

https://juejin.cn/post/6997644572629139492

最后

“在看和转发”就是最大的支持



推荐阅读
  • Linux如何安装Mongodb的详细步骤和注意事项
    本文介绍了Linux如何安装Mongodb的详细步骤和注意事项,同时介绍了Mongodb的特点和优势。Mongodb是一个开源的数据库,适用于各种规模的企业和各类应用程序。它具有灵活的数据模式和高性能的数据读写操作,能够提高企业的敏捷性和可扩展性。文章还提供了Mongodb的下载安装包地址。 ... [详细]
  • 本文介绍了在CentOS上安装Python2.7.2的详细步骤,包括下载、解压、编译和安装等操作。同时提供了一些注意事项,以及测试安装是否成功的方法。 ... [详细]
  • 本文记录了在vue cli 3.x中移除console的一些采坑经验,通过使用uglifyjs-webpack-plugin插件,在vue.config.js中进行相关配置,包括设置minimizer、UglifyJsPlugin和compress等参数,最终成功移除了console。同时,还包括了一些可能出现的报错情况和解决方法。 ... [详细]
  • 先看看ElementUI里关于el-table的template数据结构:<template><el-table:datatableData><e ... [详细]
  • Nginx使用AWStats日志分析的步骤及注意事项
    本文介绍了在Centos7操作系统上使用Nginx和AWStats进行日志分析的步骤和注意事项。通过AWStats可以统计网站的访问量、IP地址、操作系统、浏览器等信息,并提供精确到每月、每日、每小时的数据。在部署AWStats之前需要确认服务器上已经安装了Perl环境,并进行DNS解析。 ... [详细]
  • 一、Hadoop来历Hadoop的思想来源于Google在做搜索引擎的时候出现一个很大的问题就是这么多网页我如何才能以最快的速度来搜索到,由于这个问题Google发明 ... [详细]
  • 本文介绍了在rhel5.5操作系统下搭建网关+LAMP+postfix+dhcp的步骤和配置方法。通过配置dhcp自动分配ip、实现外网访问公司网站、内网收发邮件、内网上网以及SNAT转换等功能。详细介绍了安装dhcp和配置相关文件的步骤,并提供了相关的命令和配置示例。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了如何在不使用SearchBar display controller的情况下,单独使用SearchBar并捕获其textChange事件。作者介绍了实际状况,即左侧SliderMenu中的SearchBar需要在主页TableView中显示搜索结果。然后,作者提供了解决方案和步骤,帮助读者实现这一功能。 ... [详细]
  • 本文介绍了Oracle数据库中tnsnames.ora文件的作用和配置方法。tnsnames.ora文件在数据库启动过程中会被读取,用于解析LOCAL_LISTENER,并且与侦听无关。文章还提供了配置LOCAL_LISTENER和1522端口的示例,并展示了listener.ora文件的内容。 ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
  • 本文介绍了将mysql从5.6.15升级到5.7.15的详细步骤,包括关闭访问、备份旧库、备份权限、配置文件备份、关闭旧数据库、安装二进制、替换配置文件以及启动新数据库等操作。 ... [详细]
  • 本文介绍了如何清除Eclipse中SVN用户的设置。首先需要查看使用的SVN接口,然后根据接口类型找到相应的目录并删除相关文件。最后使用SVN更新或提交来应用更改。 ... [详细]
  • 本文介绍了在wepy中运用小顺序页面受权的计划,包含了用户点击作废后的从新受权计划。 ... [详细]
author-avatar
hitwill
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有