vue实现购物车功能

实现功能

CSS部分

   <style>.tr {display: flex;}.th {margin: 10px;width: 20%;height: 50%;}.td {display: flex;margin: 10px;width: 20%;height: 100px;align-items: center;}.app-container .banner-box {border-radius: 20px;overflow: hidden;margin-bottom: 10px;}.app-container .banner-box img {width: 100%;}.app-container .nav-box {background: #ddedec;height: 60px;border-radius: 10px;padding-left: 20px;display: flex;align-items: center;}.app-container .nav-box .my-nav {display: inline-block;background: #5fca71;border-radius: 5px;width: 90px;height: 35px;color: white;text-align: center;line-height: 35px;margin-right: 10px;}.breadcrumb {font-size: 16px;color: gray;}.table {width: 100%;text-align: left;border-radius: 2px 2px 0 0;border-collapse: separate;border-spacing: 0;}.table img {width: 100px;height: 100px;}button {outline: 0;box-shadow: none;color: #fff;background: #d9363e;border-color: #d9363e;color: #fff;background: #d9363e;border-color: #d9363e;line-height: 1.5715;position: relative;display: inline-block;font-weight: 400;white-space: nowrap;text-align: center;background-image: none;border: 1px solid transparent;box-shadow: 0 2px 0 rgb(0 0 0 / 2%);cursor: pointer;transition: all 0.3s cubic-bezier(0.645, 0.045, 0.355, 1);-webkit-user-select: none;-moz-user-select: none;-ms-user-select: none;user-select: none;touch-action: manipulation;height: 32px;padding: 4px 15px;font-size: 14px;border-radius: 2px;}button.pay {background-color: #3f85ed;margin-left: 20px;}.bottom {height: 60px;display: flex;align-items: center;justify-content: space-between;padding-right: 20px;border: 1px solid #f0f0f0;border-top: none;padding-left: 20px;}.right-box {display: flex;align-items: center;}.empty {text-align: center;font-size: 30px;}/* 选中时颜色是灰色 */.tr.active {background-color: #f5f7fa;}</style>

HTML

<body>
<div class="app-container" id="app"><!-- 顶部banner --><div class="banner-box"><img src="./fruitPot/封面.png" alt=""/></div><!-- 面包屑 --><div class="breadcrumb"><span>🏠</span><span>购物车</span></div><!-- 购物车主体 --><div class="main" v-if="fruitlist.length>0"><div class="table"><!-- 头部 --><div class="thead"><div class="tr"><div class="th">选中</div><div class="th th-pic">图片</div><div class="th">单价</div><div class="th num-th">个数</div><div class="th">小计</div><div class="th">操作</div></div></div><!-- 身体 --><div class="tbody"><div class="tr" :class="{active:item.isChecked}" v-for="(item) in fruitlist" :key="item.id"><div class="td"><input type="checkbox" v-model="item.isChecked"/></div><div class="td pot"><img :src="item.icon" alt=""/></div><div class="td">{{ item.price }}</div><div class="td"><div class="my-input-number"><!--disabled:禁用  --><button :disabled="item.num <=0 " class="decrease" @click="sub(item.id)"> -</button><span class="my-input__inner">{{ item.num }}</span><button class="increase" @click="add(item.id)"> +</button></div></div><div class="td">{{item.price * item.num}}</div><!-- 删除filter --><div class="td"><button @click="del(item.id)">删除</button></div></div></div></div><!-- 底部 --><div class="bottom"><!-- 全选 --><label class="check-all"><input type="checkbox" v-model="isAll"/>全选</label><div class="right-box"><!-- 所有商品总价 --><span class="price-box">总价&nbsp;&nbsp;:&nbsp;&nbsp;¥&nbsp;<span class="price">{{totalPrice()}}</span></span><!-- 结算按钮 --><button class="pay">结算( {{totalNum()}} )</button></div></div></div><!-- 空车 --><div class="empty" v-else>🛒空空如也</div>
</div>

JS

<script><!--    初始化-->// const defaultArr =[ ]const app = new Vue({el: '#app',data: {// // 水果列表从本地读取// // JSON.parse将本地JSON格式转回去// 加||defaultArr是怕用户删除完,下次刷新后只剩下空空如也// fruitlist: JSON.parse(localStorage.getItem('list'))||defaultArr,fruitlist: [{id: 1,icon: './fruitPot/苹果.png',isChecked: true,num: 2,price: 6,},{id: 2,icon: './fruitPot/苹果.png',isChecked: false,num: 7,price: 20,},{id: 3,icon: './fruitPot/鸭梨.png',isChecked: true,num: 2,price: 6,},],},// 计算属性computed: {isAll:{// isAll:  对象// get()   方法get(){// 必须所有的小选框都选中,全选按钮才选中--every// 此时的item与上面的item没关联,只是一个形参return this.fruitlist.every(item => {return  item.isChecked===true})},// value是复选框的布尔值set(value){//console.log(value)  ture/false//基于拿到的布尔值,要让所有的小选框同步状态this.fruitlist.forEach(item =>item.isChecked = value)}}},methods: {del(id) {this.fruitlist = this.fruitlist.filter(item => item.id !== id)},add(id) {// 1.根据id 找到数组中的对应项 --findconst fruit = this.fruitlist.find(item => item.id === id)// 2.操作num数量fruit.num++},sub(id) {const fruit = this.fruitlist.find(item => item.id === id)fruit.num--},//     总数量  reducetotalNum(){return this.fruitlist.reduce((sum, item) => {if (item.isChecked) {// 选中 → 需要累加return sum + item.num} else {// 没选中 → 不需要累加return sum}}, 0)},// 总价totalPrice(){return this.fruitlist.reduce((sum, item) => {if (item.isChecked) {return sum + item.num * item.price} else {return sum}}, 0)},},//     监视数据变化watch:{fruitlist:{deep:true,handler(newValue){console.log(newValue)//     需要将变化后的newValue存入本地(转JSON)localStorage.setItem('list',JSON.stringify(newValue))}}}})
</script>

全部代码

<!-- 标签\watch\methods -->
<!DOCTYPE html>
<html>
<head><meta charset="utf-8"><title>水果购物车</title><style>.tr {display: flex;}.th {margin: 10px;width: 20%;height: 50%;}.td {display: flex;margin: 10px;width: 20%;height: 100px;align-items: center;}.app-container .banner-box {border-radius: 20px;overflow: hidden;margin-bottom: 10px;}.app-container .banner-box img {width: 100%;}.app-container .nav-box {background: #ddedec;height: 60px;border-radius: 10px;padding-left: 20px;display: flex;align-items: center;}.app-container .nav-box .my-nav {display: inline-block;background: #5fca71;border-radius: 5px;width: 90px;height: 35px;color: white;text-align: center;line-height: 35px;margin-right: 10px;}.breadcrumb {font-size: 16px;color: gray;}.table {width: 100%;text-align: left;border-radius: 2px 2px 0 0;border-collapse: separate;border-spacing: 0;}.table img {width: 100px;height: 100px;}button {outline: 0;box-shadow: none;color: #fff;background: #d9363e;border-color: #d9363e;color: #fff;background: #d9363e;border-color: #d9363e;line-height: 1.5715;position: relative;display: inline-block;font-weight: 400;white-space: nowrap;text-align: center;background-image: none;border: 1px solid transparent;box-shadow: 0 2px 0 rgb(0 0 0 / 2%);cursor: pointer;transition: all 0.3s cubic-bezier(0.645, 0.045, 0.355, 1);-webkit-user-select: none;-moz-user-select: none;-ms-user-select: none;user-select: none;touch-action: manipulation;height: 32px;padding: 4px 15px;font-size: 14px;border-radius: 2px;}button.pay {background-color: #3f85ed;margin-left: 20px;}.bottom {height: 60px;display: flex;align-items: center;justify-content: space-between;padding-right: 20px;border: 1px solid #f0f0f0;border-top: none;padding-left: 20px;}.right-box {display: flex;align-items: center;}.empty {text-align: center;font-size: 30px;}/* 选中时颜色是灰色 */.tr.active {background-color: #f5f7fa;}</style>
</head>
<body>
<div class="app-container" id="app"><!-- 顶部banner --><div class="banner-box"><img src="./fruitPot/封面.png" alt=""/></div><!-- 面包屑 --><div class="breadcrumb"><span>🏠</span><span>购物车</span></div><!-- 购物车主体 --><div class="main" v-if="fruitlist.length>0"><div class="table"><!-- 头部 --><div class="thead"><div class="tr"><div class="th">选中</div><div class="th th-pic">图片</div><div class="th">单价</div><div class="th num-th">个数</div><div class="th">小计</div><div class="th">操作</div></div></div><!-- 身体 --><div class="tbody"><div class="tr" :class="{active:item.isChecked}" v-for="(item) in fruitlist" :key="item.id"><div class="td"><input type="checkbox" v-model="item.isChecked"/></div><div class="td pot"><img :src="item.icon" alt=""/></div><div class="td">{{ item.price }}</div><div class="td"><div class="my-input-number"><!--disabled:禁用  --><button :disabled="item.num <=0 " class="decrease" @click="sub(item.id)"> -</button><span class="my-input__inner">{{ item.num }}</span><button class="increase" @click="add(item.id)"> +</button></div></div><div class="td">{{item.price * item.num}}</div><!-- 删除filter --><div class="td"><button @click="del(item.id)">删除</button></div></div></div></div><!-- 底部 --><div class="bottom"><!-- 全选 --><label class="check-all"><input type="checkbox" v-model="isAll"/>全选</label><div class="right-box"><!-- 所有商品总价 --><span class="price-box">总价&nbsp;&nbsp;:&nbsp;&nbsp;¥&nbsp;<span class="price">{{totalPrice()}}</span></span><!-- 结算按钮 --><button class="pay">结算( {{totalNum()}} )</button></div></div></div><!-- 空车 --><div class="empty" v-else>🛒空空如也</div>
</div><script src="https://cdn.jsdelivr.net/npm/vue@2.7.10/dist/vue.js"></script><script><!--    初始化-->// const defaultArr =[ ]const app = new Vue({el: '#app',data: {// // 水果列表从本地读取// // JSON.parse将本地JSON格式转回去// 加||defaultArr是怕用户删除完,下次刷新后只剩下空空如也// fruitlist: JSON.parse(localStorage.getItem('list'))||defaultArr,fruitlist: [{id: 1,icon: './fruitPot/苹果.png',isChecked: true,num: 2,price: 6,},{id: 2,icon: './fruitPot/苹果.png',isChecked: false,num: 7,price: 20,},{id: 3,icon: './fruitPot/鸭梨.png',isChecked: true,num: 2,price: 6,},],},// 计算属性computed: {isAll:{// isAll:  对象// get()   方法get(){// 必须所有的小选框都选中,全选按钮才选中--every// 此时的item与上面的item没关联,只是一个形参return this.fruitlist.every(item => {return  item.isChecked===true})},// value是复选框的布尔值set(value){//console.log(value)  ture/false//基于拿到的布尔值,要让所有的小选框同步状态this.fruitlist.forEach(item =>item.isChecked = value)}}},methods: {del(id) {this.fruitlist = this.fruitlist.filter(item => item.id !== id)},add(id) {// 1.根据id 找到数组中的对应项 --findconst fruit = this.fruitlist.find(item => item.id === id)// 2.操作num数量fruit.num++},sub(id) {const fruit = this.fruitlist.find(item => item.id === id)fruit.num--},//     总数量  reducetotalNum(){return this.fruitlist.reduce((sum, item) => {if (item.isChecked) {// 选中 → 需要累加return sum + item.num} else {// 没选中 → 不需要累加return sum}}, 0)},// 总价totalPrice(){return this.fruitlist.reduce((sum, item) => {if (item.isChecked) {return sum + item.num * item.price} else {return sum}}, 0)},},//     监视数据变化watch:{fruitlist:{deep:true,handler(newValue){console.log(newValue)//     需要将变化后的newValue存入本地(转JSON)localStorage.setItem('list',JSON.stringify(newValue))}}}})
</script>
</body>
</html>

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

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

相关文章

input中文输入法导致的高频事件

这是基本结构 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><title>中文输入法的高频事件</title&…

通信-CAN-01 总线拓扑

本文主要介绍CAN总线拓扑&#xff0c;并结合实际用到CAN设备做些说明。 1 总线拓扑 拓扑结构中分为CPU&#xff0c;CAN 控制器&#xff0c;收发器&#xff0c;双绞线。CAN控制器根据两根线上的电位差来判断总线电平。发送方通过使总线发生变化&#xff0c;将消息发送给接收方…

BPSK调制解调

BPSK数字调制是相移键控PSK的一种&#xff0c;通过数字信号&#xff0c;调制载波的相位&#xff0c;利用载波的相位变化来反映数字信号&#xff0c;载波的振幅和频率均不变化。PSK应用很广泛&#xff0c;抗噪声性能比ASK和FSK要好&#xff0c;频带利用率较高。BPSK中&#xff0…

前端文件流、切片下载和上传

前端文件流、切片下载和上传技术在提升文件传输效率和优化用户体验方面发挥着关键作用。这些技术不仅可以帮助解决大文件传输过程中可能遇到的问题&#xff0c;如网络超时、内存溢出等&#xff0c;还能通过并行传输和断点续传等功能&#xff0c;提高传输速度和稳定性。 一、前端…

每日学习笔记:C++ 11的Tuple

#include <tuple> Tuple介绍(不定数的值组--可理解为pair的升级版) 定义 创建 取值 初始化 获取tuple元素个数、获取tuple某元素类型、将2个tuple类型串接为1个新tuple类型

解决Ubuntu 16.04/18.04 图形化界面异常、鼠标光标消失、鼠标变成叉叉等问题

bug场景&#xff1a; 一切从一次换源说起…叭叭叭 这篇文章解决的问题&#xff1a; 1.换源&#xff0c;默认源太慢&#xff0c;换成可用的阿里云的源 2.apt-get failed to …问题 3.图形化异常问题 4.get unmet dependence 问题 5. 鼠标光标消失和鼠标变成叉叉问题。 解决方…

【Python】time模块

专栏文章索引&#xff1a;Python 目录 一、介绍​编辑 二、常用函数​编辑 一、介绍 Python 的 time 模块提供了处理时间的函数。 二、常用函数 1.time()&#xff1a;返回当前时间的时间戳&#xff08;从1970年1月1日开始计时的秒数&#xff09;。 import timecurrent_ti…

Android Gradle 开发与应用 (五) : 基于Gradle 8.2,创建Gradle插件

1. 前言 本文介绍在Android中&#xff0c;如何基于Gradle 8.2&#xff0c;创建Gradle插件。 1.1 本文环境 Android Studio 版本 : Android Studio Hedgehog | 2023.1.1Gralde版本 : gradle 8.2 使用 Android Gradle 插件升级助理 Android Gradle 插件版本说明 1.2 为什么要写…

蓝桥杯递推与递归法|斐波那契数列|数字三角形|42点问题|数的计算|数的划分(C++)

递归是用来做dfs&#xff0c;是搜索算法的基础 递推是用来做dp部分&#xff0c;及部分其他算法&#xff0c;复杂度较低&#xff0c;不会出现爆栈问题递推法&#xff1a; 递推法是一种在数学和其他领域广泛应用的重要方法&#xff0c;它在计算机科学中被用作一种关键的数值求解…

LLM 推理优化探微 (3) :如何有效控制 KV 缓存的内存占用,优化推理速度?

编者按&#xff1a; 随着 LLM 赋能越来越多需要实时决策和响应的应用场景&#xff0c;以及用户体验不佳、成本过高、资源受限等问题的出现&#xff0c;大模型高效推理已成为一个重要的研究课题。为此&#xff0c;Baihai IDP 推出 Pierre Lienhart 的系列文章&#xff0c;从多个…

C/C++指针详解

接下来我们来介绍一下什么是指针&#xff1f; 指针其实就是元素存放地址&#xff0c;更加形象的比喻&#xff1a;在酒店中如果你想要去注必须去付费不然不能住&#xff0c;在计算机也同样如此&#xff08;但是不需要付费哦&#xff09;每当我们使用一个变量或其他需要申请空间…

Linux centos6安装rz、sz命令

centos6传文件提示command not found # yum install lrzsz 提示错误 wget http://www.ohse.de/uwe/releases/lrzsz-0.12.20.tar.gz 下载离线包 https://www.ohse.de/uwe/software/lrzsz.html 下载最新版本 [rootnode1 ~]# tar -zxvf lrzsz-0.12.20.tar.gz …

天啊,腾讯云轻量服务器流量用完了怎么办?

腾讯云轻量服务器流量用完了怎么办&#xff1f;超额部分的流量另外支付流量费&#xff0c;流量价格为0.8元/GB&#xff0c;会自动扣你的腾讯云余额&#xff0c;如果你的腾讯云账号余额不足&#xff0c;那么你的轻量应用服务器会面临停机&#xff0c;停机后外网无法访问&#xf…

RAG综述 《Retrieval-Augmented Generation for Large Language Models: A Survey》笔记

文章目录 概述RAG 的定义RAG的框架Naive RAGAdvanced RAGpre-retrieval processRetrievalpost-retrieval process Modular RAG RetrievalEnhancing Semantic Representationschunk 优化 微调向量模型Aligning Queries and DocumentsAligning Retriever and LLM GenerationAugme…

ES分页查询的最佳实践:三种方案

Elasticsearch&#xff08;ES&#xff09;中进行分页查询时&#xff0c;最佳实践取决于具体的使用场景和需求。 以下是对每种分页方法的简要分析以及它们适用的情况&#xff1a; 1. From Size 最常见且直观的方法&#xff0c;通过from参数指定跳过多少条记录&#xff0c;si…

07 数据结构之图

# Makefile CCgcc CFLAGS -g -Wall SRCStest.c graph.c link_queue.c OBJS$(SRCS:.c.o) #variable replace APPtestall:$(OBJS) #指定一个目标&#xff0c; 不然默认目标不会检查依赖文件的时间戳$(CC) $(SRCS) -o $(APP) .PH…

MySQL中常用的操作语句已汇总

目录 一、库语句 1.查询现有数据库 2.创建数据库 3.选中数据库 ​编辑 4.删除数据库 二、初阶表操作 1.查看数据库现有表 2.查看表结构 3.创建表 4.删除表 5.全列查询 6.删除表2 7.修改操作 三、插入操作 1.全列插入 2.指定列插入 3.一次插入多组数据 4.插入…

【AI视野·今日NLP 自然语言处理论文速览 第八十二期】Tue, 5 Mar 2024

AI视野今日CS.NLP 自然语言处理论文速览 Tue, 5 Mar 2024 (showing first 100 of 175 entries) Totally 100 papers &#x1f449;上期速览✈更多精彩请移步主页 Daily Computation and Language Papers Key-Point-Driven Data Synthesis with its Enhancement on Mathematica…

微服务---Eureka注册中心

目录 一、服务中的提供者与消费者 二、Eureka工作流程 三、搭建Eureka服务 四、服务拉取 五、总结 1.搭建EurekaServer 2.服务注册 3.服务发现 一、服务中的提供者与消费者 服务提供者&#xff1a;一次业务中&#xff0c;被其他微服务调用的服务。即提供接口给其他微服务。…

Golang + Redis解决缓存击穿(双层缓存)

Golang Redis解决缓存穿透&#xff08;双层缓存&#xff09; 代码地址&#xff1a; https://github.com/ziyifast/ziyifast-code_instruction/tree/main/redis_demo/cache_breakdown 1 概念 目前主流方案是在数据库前加一层缓存。类似于DB的防弹衣。 缓存击穿&#xff1a;Re…