微信小程序实战案例 - 餐馆点餐系统 阶段 2 – 购物车

news2025/5/15 19:24:08

阶段 2 – 购物车(超详细版)

目标

  1. 把“加入购物车”做成 全局状态,任何页面都能读写
  2. 在本地 持久化(关闭小程序后购物车仍在)
  3. 新建 购物车页:数量增减、总价实时计算、去结算入口
  4. 打 Git Tag v2.0‑cart

1. 学到的核心技术

技术/概念关键 API/组件为什么要学
全局状态App.globalData / 简易 store.js小项目先不引入第三方状态库,足够用
本地持久化wx.setStorageSync / wx.getStorageSync保证用户切后台、重进小程序后数据不丢
UI 组件(弹出购物车)
 (底部结算条)
 (数量加减)
快速完成电商式交互
TabBar Badgewx.setTabBarBadge在底部“购物车”图标上显示件数

2. 项目结构新增

miniprogram/
  ├─ store/          # 新增
  │   └─ cart.js
  ├─ pages/
  │   ├─ index/      # 首页已存在
  │   └─ cart/       # 新建购物车页面

3. 编写轻量级全局 Store

路径miniprogram/store/cart.js

const CART_KEY = 'CART_V1'
const store = {
  data: { items: {} },

  load() {
    this.data.items = wx.getStorageSync(CART_KEY) || {}
  },
  save() {
    wx.setStorageSync(CART_KEY, this.data.items)
  },
  add(dish) {
    const { _id } = dish
    if (this.data.items[_id]) {
      this.data.items[_id].count += 1
    } else {
      this.data.items[_id] = { ...dish, count: 1 }
    }
    this.save()
  },
  totalCount() {
    return Object.values(this.data.items).reduce((s, i) => s + i.count, 0)
  },
  totalPrice() {
    return Object.values(this.data.items)
      .reduce((s, i) => s + i.count * i.price, 0)
      .toFixed(2)
  }
  
}
module.exports = store


4. 在 app.js 中加载购物车

// miniprogram/app.js
App({
  onLaunch() {
    const cart = require('./store/cart')
    cart.load()
    this.globalData = { cart }
  }
})

5. 修改首页:加入购物车 & 更新角标

文件pages/index/index.js

只需要把之前的 onAddCart 改成:

import cart from '../../store/cart'

onAddCart(e) {
    const dish = e.currentTarget.dataset.dish
    cart.add(dish)
    wx.setTabBarBadge({ index: 1, text: String(cart.totalCount()) })
    wx.showToast({ title: '已加入购物车', icon: 'success' })
  }
}

提示:在 app.jsontabBar 数组里,把第二项页面路径设成 "pages/cart/index",这样角标才会显示在购物车图标上。


6. 新建购物车页面

6.1 组件声明 pages/cart/index.json
{
  "navigationBarTitleText": "购物车"
}
6.2 页面布局 index.wxml
<view class="page">
  <block wx:for="{{list}}" wx:key="_id">
    <view class="cart-item">
      <image class="thumb" src="{{item.img}}" mode="aspectFill" />
      <view class="info">
        <text class="name">{{item.name}}</text>
        <text class="price">¥{{item.price}}</text>
        <text class="count">x {{item.count}}</text>
      </view>
    </view>
  </block>

  <view class="bottom-bar">
    <text>共 {{totalCount}} 件</text>
    <text>合计:¥{{totalPrice}}</text>
    <button type="primary" bindtap="onCheckout">去结算</button>
  </view>
</view>

6.3 页面逻辑 index.js
const cart = require('../../store/cart')

Page({
  data: {
    list: [],
    totalCount: 0,
    totalPrice: '0.00'
  },

  onShow() {
    cart.load()
    this.refresh()
  },

  refresh() {
    const items = Object.values(cart.data.items)
    const totalCount = cart.totalCount()
    const totalPrice = cart.totalPrice()

    this.setData({
      list: items,
      totalCount,
      totalPrice
    })
  },

  onCheckout() {
    if (!this.data.totalCount) {
      wx.showToast({ title: '购物车为空', icon: 'none' })
      return
    }
    wx.navigateTo({ url: '/pages/confirm/index' }) // 下一阶段页
  }
})


6.4 简单样式 index.wxss
.page {
  padding: 20rpx;
}

.cart-item {
  display: flex;
  background: #fff;
  padding: 20rpx;
  margin-bottom: 20rpx;
  border-radius: 16rpx;
  box-shadow: 0 4rpx 12rpx rgba(0,0,0,0.05);
}

.thumb {
  width: 100rpx;
  height: 100rpx;
  border-radius: 8rpx;
  margin-right: 20rpx;
}

.info {
  display: flex;
  flex-direction: column;
  justify-content: space-around;
}

.name {
  font-weight: bold;
  font-size: 32rpx;
}

.price {
  color: #fa541c;
}

.count {
  font-size: 28rpx;
  color: #888;
}

.bottom-bar {
  position: fixed;
  bottom: 0;
  left: 0;
  width: 100%;
  background: #fff;
  padding: 20rpx;
  display: flex;
  justify-content: space-between;
  box-shadow: 0 -2rpx 12rpx rgba(0,0,0,0.1);
}


7. TabBar 配置示例

// app.json(节选)
"tabBar": {
    "list": [
      { "pagePath": "pages/index/index", "text": "菜单", "iconPath": "images/icons/home.png", "selectedIconPath": "images/icons/home-active.png" },
      { "pagePath": "pages/cart/index",  "text": "购物车", "iconPath": "images/icons/business.png", "selectedIconPath": "images/icons/business-active.png" }
    ]
  }

在这里插入图片描述

8. 自测清单 & Git Tag

  1. 首页点两道菜 → 角标显示 2
  2. 进入购物车页
    • 看得到两条记录
  3. 关闭小程序再打开 → 数据依旧存在
  4. 一切通过后:
git add .
git commit -m "feat: shopping cart"
git tag v2.0-cart
git push --tags

9. 练习(进阶挑战)

难度练习内容
cart.js 增加 clear() 方法,在购物车页提供“一键清空”。
⭐⭐在首页卡片上显示当前已选数量(小圆角徽标)。
⭐⭐⭐把 Store 升级为 PiniaRemax Recoil,体验响应式自动刷新。

阶段小结

  • 你已拥有 加入购物车 → 全局状态 → 本地持久化 → 购物车 UI 的完整链路。
  • 代码量 ≈ 250 行,但逻辑清晰、易维护。
  • 接下来进入 阶段 3 – 下单 & 云数据库:把购物车内容写入 orders 集合,实现真正的下单流程。

继续加油,愉快编码!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2334685.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

sql 向Java的映射

优化建议&#xff0c;可以在SQL中控制它的类型 在 MyBatis 中&#xff0c;如果返回值类型设置为 java.util.Map&#xff0c;默认情况下可以返回 多行多列的数据

Visual Studio未能加载相应的Package包弹窗报错

环境介绍&#xff1a; visulal studio 2019 问题描述&#xff1a; 起因&#xff1a;安装vs扩展插件后&#xff0c;重新打开Visual Studio&#xff0c;报了一些列如下的弹窗错误&#xff0c;即使选择不继续显示该错误&#xff0c;再次打开后任然报错&#xff1b; 解决思路&am…

【HD-RK3576-PI】Docker搭建与使用

硬件&#xff1a;HD-RK3576-PI 软件&#xff1a;Linux6.1Ubuntu22.04 1.Docker 简介 Docker 是一个开源的应用容器引擎&#xff0c;基于 Go 语言开发&#xff0c;遵循 Apache 2.0 协议。它可以让开发者将应用程序及其依赖项打包到一个轻量级、可移植的容器中&#xff0c;并在任…

【websocket】使用案例( ​JSR 356 标准)

目录 一、JSR 356方式&#xff1a;简单示例 1、引入依赖 2、注册端点扫描器 3、编写通过注解处理生命周期和消息 4、细节解读 5、总结 二、聊天室案例 方案流程 1、引入依赖 2、注册端点扫描器 3、编写一个配置类&#xff0c;读取httpsession 4、编写通过注解处理生…

IS-IS中特殊字段——OL过载

文章目录 OL 过载位 &#x1f3e1;作者主页&#xff1a;点击&#xff01; &#x1f916;Datacom专栏&#xff1a;点击&#xff01; ⏰️创作时间&#xff1a;2025年04月13日20点12分 OL 过载位 路由过载 使用 IS-IS 的过载标记来标识过载状态 对设备设置过载标记后&#xff…

【时频谱分析】快速谱峭度

算法配置页面&#xff0c;也可以一键导出结果数据 报表自定义绘制 获取和下载【PHM学习软件PHM源码】的方式 获取方式&#xff1a;Docshttps://jcn362s9p4t8.feishu.cn/wiki/A0NXwPxY3ie1cGkOy08cru6vnvc

Spring Boot 支持的内嵌服务器(Tomcat、Jetty、Undertow、Netty(用于 WebFlux 响应式应用))详解

Spring Boot 支持的内嵌服务器详解 1. 支持的内嵌服务器 Spring Boot 默认支持以下内嵌服务器&#xff1a; Tomcat&#xff08;默认&#xff09;JettyUndertowNetty&#xff08;用于 WebFlux 响应式应用&#xff09; 2. 各服务器使用示例 (1) Tomcat&#xff08;默认&#xf…

微软Exchange管理中心全球范围宕机

微软已确认Exchange管理中心&#xff08;Exchange Admin Center&#xff0c;EAC&#xff09;发生全球性服务中断&#xff0c;导致管理员无法访问关键管理工具。该故障被标记为关键服务事件&#xff08;编号EX1051697&#xff09;&#xff0c;对依赖Exchange Online的企业造成广…

基于Qt的串口通信工具

程序介绍 该程序是一个基于Qt的串口通信工具&#xff0c;专用于ESP8266 WiFi模块的AT指令配置与调试。主要功能包括&#xff1a; 1. 核心功能 串口通信&#xff1a;支持串口开关、参数配置&#xff08;波特率、数据位、停止位、校验位&#xff09;及数据收发。 AT指令操作&a…

html简易实现推箱子小游戏原理(易上手)

实现效果 使用方向键移动&#xff0c;将橙色箱子推到绿色目标区域&#xff08;黑色块为墙&#xff0c;白色块为可通过区域&#xff0c;蓝球为小人&#xff09; 实现过程 <!DOCTYPE html> <html> <head><title>推箱子小游戏</title><style&g…

字符串与栈和队列-算法小结

字符串 双指针 反转字符串(双指针) 力扣题目链接 void reverseString(vector<char>& s) {for (int i 0, j s.size() - 1; i < s.size()/2; i, j--) {swap(s[i],s[j]);} }反转字符串II 力扣题目链接 遍历字符串的过程中&#xff0c;只要让 i (2 * k)&#…

类似东郊到家的上门按摩预约服务系统小程序APP源码全开源

&#x1f525; 为什么上门按摩正在席卷全国&#xff1f; 万亿蓝海市场爆发 2024年中国按摩市场规模突破8000亿&#xff0c;上门服务增速达65% 90后成消费主力&#xff0c;**72%**白领每月至少使用1次上门按摩&#xff08;数据来源&#xff1a;艾媒咨询&#xff09; 传统痛点…

Python | 在Pandas中按照中值对箱形图排序

箱形图是可视化数据分布的强大工具&#xff0c;因为它们提供了对数据集内的散布、四分位数和离群值的洞察。然而&#xff0c;当处理多个组或类别时&#xff0c;通过特定的测量&#xff08;如中位数&#xff09;对箱形图进行排序可以提高清晰度并有助于揭示模式。在本文中&#…

游戏引擎学习第215天

总结并为今天做铺垫 今天的工作内容是解决调试系统中的一个小问题。昨天我们已经完成了大部分的调试系统工作&#xff0c;但还有一个小部分没有完全处理&#xff0c;那就是关于如何层次化组织数据的问题。我们遇到的一个问题是&#xff0c;演示代码中仍有一个尚未解决的部分&a…

【Redis】redis事物与管道

Redis 事务&#xff08;Transaction&#xff09; 事务概念 事务&#xff1a;是一组操作的集合&#xff0c;是不可分割的工作单元。Redis 事务特点&#xff1a; 一个事务可以一次执行多个命令。所有命令都被顺序化&#xff0c;形成一个队列。所有命令在执行 EXEC 时一次性、顺…

Django信号使用完全指南示例

推荐超级课程: 本地离线DeepSeek AI方案部署实战教程【完全版】Docker快速入门到精通Kubernetes入门到大师通关课AWS云服务快速入门实战目录 **引言:****先决条件:****目录:****1. 什么是Django信号?****2:设置你的Django项目****2.1. 安装Django**2.2. 创建一个Django项…

vulkanscenegraph显示倾斜模型(5.9)-vsg中vulkan资源的编译

前言 上一章深入剖析了GPU资源内存及其管理&#xff0c;vsg中为了提高设备内存的利用率&#xff0c;同时减少内存(GPU)碎片&#xff0c;采用GPU资源内存池机制(vsg::MemoryBufferPools)管理逻辑缓存(VkBuffer)与物理内存(VkDeviceMemory)。本章将深入vsg中vulkan资源的编译(包含…

今日行情明日机会——20250409

今日行情还需要考虑关税对抗~ 2025年4月8日涨停的主要行业方向分析 1. 军工&#xff08;19家涨停&#xff09; 细分领域&#xff1a;国防装备、航空航天、军民融合。催化因素&#xff1a;国家安全战略升级、国防预算增加、重大军工项目落地预期。 2. 免税&#xff08;15家涨…

系统分析师(六)-- 计算机网络

概述 TCP/IP 协议族 DNS DHCP 网络规划与设计 逻辑网络设计 物理网络设计 题目 层次化网络设计 网络冗余设计 综合布线系统 IP地址 网络接入技术 其他网络技术应用 物联网

Qwen2.5-VL Technical Report 论文翻译和理解

一、TL&#xff1b;DR Qwen2.5-VL是QwenVL的最新模型&#xff0c;在视觉识别、精准目标定位、稳健文档解析以及长视频理解等方面实现了重大突破引入了动态分辨率处理和绝对时间编码&#xff0c;使其能够处理不同尺寸的图像以及长达数小时的视频&#xff0c;并实现秒级事件定位…