文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

【毕业设计】 微信小程序购物商城系统 【含代码】

2023-10-27 05:19

关注

文章目录

0 前言

Hi,同学们好呀,学长今天带大家做一做小程序开发

仿得物微信小程序

1 开发工具

2 总体架构

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

项目中我负责的部分主要如下(一些数据为固定数据写在config中,js文件通过module.exports暴露,需要引用时在页面对应js头部引入,例const {} = require(‘…/…/…/…/config/buys’))。项目中我使用的较多vant组件,需要在构建npm包时引入vant,详情可见有赞vant的npm安装。页面使用第三方组件时须在对应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  搜索页

3 项目规划

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

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

在这里插入图片描述

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

4 云数据库

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

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

在这里插入图片描述
建立数据集合后需修改数据权限才可正常访问。

在这里插入图片描述

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

在这里插入图片描述

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

5 项目解构

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

在这里插入图片描述
在这里插入图片描述

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

5.1 购买首页

在这里插入图片描述

购买首页样式

在这里插入图片描述

<view class="page">  <!-- 使用van-sticky设置dewu-hd吸顶 搜索栏-->  <van-sticky>      <!-- dewu-hd使用flex布局 -->    <view class="dewu-hd">      <view class="dewu-hd-search" bindtap="gotoSearch">        <van-search placeholder="搜索单号" disabled />      </view>      <view class="dewu-kinds" bindtap="kinds"><image src=""></image>      </view>    </view>  </van-sticky>  <!-- van-tabs实现购买页导航及与内容页对应 -->  <van-tabs class="dewu-tabs">    <van-tab title="推荐">      <view class="dewu-tip">        <view class="dewu-tip-img-hd"><image src=""></image>        </view>        <!-- 使用van-grid布局设置边框隐藏快速定位 -->        <van-grid>          <van-grid-item use-slot>            <image style="" src=""></image>            <text>正品保障</text>          </van-grid-item>        </van-grid>      </view>      <view class="van-items">        <van-grid class="van-grid-bd">        <!-- grid布局自定义van-grid-item样式 -->          <van-grid-item use-slot>            <view class="item-img"><image src=""></image></view>            <view class="item-text">              <span>{{}}</span>            </view>          </van-grid-item>        </van-grid>      </view>    </van-tab>  </van-tabs></view>

商品项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时跳转到详情页且展示对应数据。

5.2 商品详情页

在这里插入图片描述

商品详情页样式

在这里插入图片描述

<view class="page"><!-- 头部 滑块及 -->  <view class="detail_hd">    <swiper class="swiper__hd">        <swiper-item class="swiper_hd"></swiper-item>    </swiper>    <view class="dots1">        <view class="{{current==index?'active':''}}"></view>    </view>    <view class="detail_hd-title">{{img.digest}}</view>    <view class="detail_hd-price">      <text id="p2">{{img.price}}</text>    </view>  </view>  <van-cell class="size" bind:click="showPopup1">    <view class="size-l">选择尺码</view>    <view class="size-r">请选择尺码</view>    <image class="ricon" style="width:26rpx;height:26rpx;" src=""></image>  </van-cell>    <!-- flex布局 每一个swiper-item包含三项 -->  <view class="detail_bd">  <swiper></swiper></view>  <van-goods-action>    <button>立即购买</button>  </van-goods-action></view>

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

在这里插入图片描述

<van-popup closeable position="bottom" custom-style="height: 75%">    <view class="detail_size-hd">      <view class="detail_size-hd-img">        <image bindtap="previewImage" mode="aspectFit" src="{{img.pic}}">        </image>      </view>      <view class="detail_size-hd-price">        <text style="font-size:25rpx;"></text>        <text wx:if="{{activeSizeIndex==-1}}">--</text>        <text wx:if="{{activeSizeIndex==index}}">{{item.price}}</text>      </view>      <view>        <image src=""></image>        <text wx:if="{{activeSizeIndex==-1}}">请选择商品</text>        <text wx:if="{{activeSizeIndex==index}}">已选 {{item.size}}</text>      </view>    </view>    <!-- 尺码布局 -->    <view class="detail_size-bd">      <van-grid square gutter="10">        <van-grid-item>          <view class="size">            <text id="p3">{{item.size}}</text>            <text id="p4">{{item.price}}</text>          </view>        </van-grid-item>      </van-grid>    </view>    <view>      <button>{{}}</button>    </view>  </van-popup>

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

在这里插入图片描述

<van-sticky sticky offset-top="{{ 180 }}">      <view class="head">        <view class="detail_produce-hd">相关推荐</view>        <view class="detail_close" bindtap="onClose2">          <image style="width:40rpx;height:40rpx;" src=""></image>        </view>      </view>    </van-sticky>

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

获取商品详情

  async onLoad(options) {  //获取对应_id的商品数据    console.log(options);    let id = options.id    console.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.flag    let index = e.currentTarget.dataset.index    if(flag==index) {         this.setData({        activeSizeIndex: -1,        flag: -1      })     }    else {      this.setData({        activeSizeIndex: index,        flag: index      })     }  },

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

5.3 搜索页

在这里插入图片描述
搜索页样式

在这里插入图片描述

<view class="page">  <view class="search">    <van-stichy>      <van-search value="{{value}}" bind:clear="onClear" placeholder="输入商品名称、货号"/>    </van-stichy>        <!-- block包装  flex布局 -->    <block wx:if="{{showHistory == true && historyList.length > 0}}">      <view class="historyContainer">        <view class="title">历史搜索<image class="delete" src=""></image>        </view>        <view class="historyList">          <view class="historyItem">            <text class="order">{{}}</text>          </view>        </view>      </view>    </block>  </view></view>

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

搜索记录

  async onSearch(e) {    // console.log(e);    if (!e.detail.trim()) {      wx.showToast({        title: '请输入商品名',      })      return    }    let {value, historyList} = this.data    if(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.setStorageSync将value存入缓存,wx.getStorageSync获取打印出来。通过indexOf方法判断value是否已经存在,是则删除historyList中的该项。

历史搜索

  async historySearch(e) {    // console.log(e);    let historyList = this.data.historyList    let value = historyList[e.currentTarget.dataset.index]    this.setData({      value,               //修改value      showHotList: 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将本地存储的历史记录删除。

5.4 品牌分类页

在这里插入图片描述
在这里插入图片描述

<view class="page">  <van-sticky>    <view class="search" bindtap="gotoSearch">    <van-search placeholder="搜索商品" input-align="center" disabled />  </view>  </van-sticky>    <view class="kinds">    <view class="hd">    <scroll-view class="scroll-view-left">      <view class="scroll-view-left-item {{activeNavIndex == index?'active': ''}}">        <text>{{}}</text>      </view>    </scroll-view>  </view>    <view class="bd">    <scroll-view>      <view>        <view class="kind-title">        <van-divider contentPosition="center">{{}}</van-divider>        </view>        <van-grid>        <van-grid-item>{{}}</van-grid-item>        </van-grid>      </view>    </scroll-view>  </view>  </view></view>

分类页面主要是使用了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,    })  },

5.5 筛选排序页

在这里插入图片描述

<view class="page">  <van-sticky>    <view class="search" bindtap="gotoSearch">      <van-search placeholder="{{titles}}" disabled />    </view>    <view class="tab">      <view wx:for="{{tabs}}" wx:key="index" data-index="{{index}}"        class="tab-item {{activeTabIndex == index?'active': ''}}" bindtap="changeItem">        <text>{{item.title}}</text>        <image style="width:26rpx;height:26rpx;" src="{{item.pic}}"></image>      </view>    </view>  </van-sticky></view>

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

在这里插入图片描述

  <van-popup>    <scroll-view class="pop" scroll-y>      <van-collapse>        <van-collapse-item title="适用人群" value="全部" name="1">        </van-collapse-item>        <van-grid column-num="3" gutter="{{ 10 }}">          <van-grid-item class="{{activeIndex1==index?'active1':''}}">{{}}</van-grid-item>        </van-grid>      </van-collapse>            <van-goods-action>        <button>重置</button>        <button>确定</button>      </van-goods-action>    </scroll-view>  </van-popup>

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

初始商品排序

  async onLoad(options) {    // console.log(options);    let title = options.title    let 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时点击价格降序排序并设置flag1,flag1时点击价格升序排序并设置flag==-1。

筛选排序

  pick(e) {    let flag = e.currentTarget.dataset.flag    let index = e.currentTarget.dataset.index    let cd = this.data.human[index].kind    if(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),这里无穷大并不会生效,目前优化是声明为常量.

6 最后

来源地址:https://blog.csdn.net/WEB_DC/article/details/128431502

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-移动开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯