Vue3电商项目实战-结算支付 3【05-结算-收货地址-添加、06-结算-收货地址-修改、07-结算-提交订单】

news/2025/3/15 3:34:14/

文章目录

    • 05-结算-收货地址-添加
    • 06-结算-收货地址-修改
    • 07-结算-提交订单


05-结算-收货地址-添加

目的:实现收货地址的添加。

大致步骤:

  • 独立组件,准备一个对话框
  • 完成表单布局
  • 完成确认添加操作

落的代码:

1.独立组件,准备一个对话框
src/views/member/pay/components/address-edit.vue 添加地址组件

<template><XtxDialog title="添加收货地址" v-model:visible="dialogVisible"><div class="address-edit">表单</div><template v-slot:footer><XtxButton type="gray" style="margin-right:20px">取消</XtxButton><XtxButton type="primary">确认</XtxButton></template></XtxDialog>
</template>
<script>
import { ref } from 'vue'
export default {name: 'AddressEdit',setup () {const dialogVisible = ref(false)// 打开函数const open = () => {dialogVisible.value = true}return { dialogVisible, open }}
}
</script>
<style scoped lang="less">
.address-edit {}
</style>

src/views/member/pay/components/checkout-address.vue 使用添加地址组件

    // 添加收货地址组件
+   const addressEdit = ref(null)
+   const openAddressEdit = () => {
+     addressEdit.value.open()
+   }return {showAddress,dialogVisible,selectedAddress,openDialog,confirmAddress,
+      addressEdit,
+      openAddressEdit}
<XtxButton @click="openAddressEdit()" class="btn">添加地址</XtxButton>

2.完成表单布局
src/views/member/pay/components/address-edit.vue 结构和样式

    <div class="xtx-form"><div class="xtx-form-item"><div class="label">收货人:</div><div class="field"><input class="input" placeholder="请输入收货人" /></div></div><div class="xtx-form-item"><div class="label">手机号:</div><div class="field"><input class="input" placeholder="请输入手机号" /></div></div><div class="xtx-form-item"><div class="label">地区:</div><div class="field"><XtxCity placeholder="请选择所在地区"/></div></div><div class="xtx-form-item"><div class="label">详细地址:</div><div class="field"><input class="input" placeholder="请输入详细地址" /></div></div><div class="xtx-form-item"><div class="label">邮政编码:</div><div class="field"><input class="input" placeholder="请输入邮政编码" /></div></div><div class="xtx-form-item"><div class="label">地址标签:</div><div class="field"><input class="input" placeholder="请输入地址标签,逗号分隔" /></div></div></div>
.xtx-dialog {:deep(.wrapper){width: 780px;.body {font-size: 14px;}}
}
.xtx-form {padding: 0;input {outline: none;&::placeholder {color: #ccc;}}
}
.xtx-city {width: 320px;:deep(.select) {height: 50px;line-height: 48px;display: flex;padding: 0 10px;justify-content: space-between;.placeholder {color: #ccc;}i {color: #ccc;font-size: 18px;}.value {font-size: 14px;}}:deep(.option) {top: 49px;}
}

src/components/library/xtx-city.vue 暴露占位文字

    placeholder: {type: String,default: '请选择配送地址'}
<span v-if="!fullName" class="placeholder">{{placeholder}}</span>

3.完成确认添加操作
src/views/member/pay/components/address-edit.vue 动态绑定表单

    // 表单数据const formData = reactive({receiver: '',contact: '',provinceCode: '',cityCode: '',countyCode: '',address: '',postalCode: '',addressTags: '',isDefault: 0,fullLocation: ''})// 选择地区const changeCty = (data) => {formData.provinceCode = data.provinceCodeformData.cityCode = data.cityCodeformData.countyCode = data.countyCodeformData.fullLocation = data.fullLocation}return { dialogVisible, open, formData, changeCty }
    <div class="xtx-form"><div class="xtx-form-item"><div class="label">收货人:</div><div class="field">
+          <input v-model="formData.receiver" class="input" placeholder="请输入收货人" /></div></div><div class="xtx-form-item"><div class="label">手机号:</div><div class="field">
+          <input v-model="formData.contact" class="input" placeholder="请输入手机号" /></div></div><div class="xtx-form-item"><div class="label">地区:</div><div class="field"><XtxCityplaceholder="请选择所在地区"
+            :fullLocation="form.fullLocation"
+            @change="changeCty"/></div></div><div class="xtx-form-item"><div class="label">详细地址:</div><div class="field">
+          <input  v-model="formData.address" class="input" placeholder="请输入详细地址" /></div></div><div class="xtx-form-item"><div class="label">邮政编码:</div><div class="field">
+          <input  v-model="formData.postalCode" class="input" placeholder="请输入邮政编码" /></div></div><div class="xtx-form-item"><div class="label">地址标签:</div><div class="field">
+          <input v-model="formData.addressTags" class="input" placeholder="请输入地址标签,逗号分隔" /></div></div></div>

src/api/order.js 接口函数

/*** 添加收货地址信息* @param {Object} address - 地址对象*/
export const addAddress = (address) => {return request('/member/address', 'post', address)
}

src/components/library/xtx-city.vue 提供占位文字属性

placeholder: {type: String,default: '请选择配送地区'}
<span class="placeholder" v-if="!fullLocation">{{placeholder}}</span>

src/views/member/pay/components/address-edit.vue 进行提交操作

    // 选择地区const changeCty = (data) => {formData.provinceCode = data.provinceCodeformData.cityCode = data.cityCodeformData.countyCode = data.countyCode
+      formData.fullLocation = data.fullLaction}
+    // 提交操作
+    const submit = () => {
+      addAddress(formData).then(data => {
+        // 添加成功
+        Message({ text: '添加收货地址成功', type: 'success' })
+        formData.id = data.result.id
+        dialogVisible.value = false
+        emit('on-success', formData)
+      })
+    }
+    return { dialogVisible, open, formData, changeCty, submit }

src/views/member/pay/components/checkout-address.vue 接受添加成功的地址对象

+    // 成功
+    const successHandler = (formData) => {
+      const json = JSON.stringify(formData)  // 需要克隆下,不然使用的是对象的引用
+      // eslint-disable-next-line vue/no-mutating-props
+      props.list.unshift(JSON.parse(json))
+    }return {showAddress,dialogVisible,selectedAddress,openDialog,confirmAddress,addressEdit,openAddressEdit,
+      successHandler}
    <!-- 添加地址 --><AddressEdit ref="addressEdit" @on-success="successHandler" />

src/views/member/pay/components/address-edit.vue 每次打开对话框清空表单

    // 打开函数const open = (form) => {dialogVisible.value = true
+      // 传人{}的时候就是清空,否则就是赋值
+      for (const key in formData) {
+        formData[key] = form[key]
+      }}

06-结算-收货地址-修改

目的:在添加收货地址组件基础之上完成修改逻辑。

大致步骤:

  • 打开对话框的时候传人当前需要修改的地址对象
  • 再添加组件open函数处,接收数据赋值给表单,修改标题。
  • 封装一个API接口函数实现修改,在提交事件中合并修改操作
  • 父组件修改数据

落的代码:

1.打开对话框的时候传人当前需需改的地址对象
src/views/member/pay/checkout-address.vue

<a @click="openAddressEdit(showAddress)" v-if="showAddress" href="javascript:;">修改地址</a>

2.再添加组件open函数处,接收数据赋值给表单,修改标题
src/views/member/pay/address-edit.vue

    const formData = reactive({
+      id: '',receiver: '',contact: '',provinceCode: '',cityCode: '',countyCode: '',
+      fullLocation: '',address: '',postalCode: '',addressTags: '',isDefault: 0})
<XtxDialog :title="(formData.id?'编辑':'添加')+'收货地址'" v-model:visible="dialogVisible">

3.封装一个API接口函数实现修改,在提交事件中合并修改操作
src/api/order.js

/*** 编辑收货地址信息* @param {Object} address - 地址对象*/
export const editAddress = (address) => {return request('/member/address', 'put', address)
}

src/views/member/pay/address-edit.vue

    // 打开对话框函数const open = (address) => {// 先填充数据 - 编辑if (address.id) {for (const key in formData) {formData[key] = address[key]}} else {// 先清空数据 - 添加for (const key in formData) {if (key !== 'isDefault') {formData[key] = ''}}}dialogVisible.value = true}
    // 提交操作const app = getCurrentInstance()const submit = () => {
+      if (formData.id) {
+        editAddress(formData).then(data => {
+          // 修改成功
+          Message(app, { text: '修改收货地址成功', type: 'success' })
+          dialogVisible.value = false
+          emit('on-success', formData)
+        })
+      } else {addAddress(formData).then(data => {// 添加成功Message(app, { text: '添加收货地址成功', type: 'success' })formData.id = data.result.iddialogVisible.value = falseemit('on-success', formData)})
+      }}

4.父组件修改数据
src/views/member/pay/components/checkout-address.vue

    // 成功const successHandler = (formData) => {
+      const editAddress = props.list.find(item => item.id === formData.id)
+      if (editAddress) {
+        // 修改
+        for (const key in editAddress) {
+          editAddress[key] = formData[key]
+        }
+      } else {// 添加const json = JSON.stringify(formData) // 需要克隆下,不然使用的是对象的引用// eslint-disable-next-line vue/no-mutating-propsprops.list.unshift(JSON.parse(json))
+      }

07-结算-提交订单

目的:汇总提交订单需要的数据,进行提交。

大致步骤:

  • 定义需要提交的数据对象
  • 绑定提交订单点击事件,进行提交即可

落的代码:

1.定义需要提交的数据对象
src/views/member/pay/checkout.vue

  setup () {const checkoutInfo = ref(null)findCheckoutInfo().then(data => {checkoutInfo.value = data.result
+      // 设置提交时候的商品
+      requestParams.goods = checkoutInfo.value.goods.map(item => {
+        return {
+          skuId: item.skuId,
+          count: item.count
+        }
+      })})// 需要提交的字段const requestParams = reactive({addressId: null,
+      deliveryTimeType: 1,
+      payType: 1,
+      buyerMessage: '',
+      goods: []})

2.绑定提交订单点击事件,进行提交即可
src/api/order.js 提交订单API函数

/*** 提交订单* @param {Object} order - 订单信息对象*/
export const createOrder = (order) => {return request('/member/order', 'post', order)
}

src/views/member/pay/checkout.vue 提交订单

        <!-- 提交订单 --><div class="submit"><XtxButton @click="submitOrder" type="primary">提交订单</XtxButton></div>
    // 提交订单const router = useRouter()const submitOrder = () => {if (!requestParams.addressId) return Message({ text: '请选择收货地址' })createOrder(requestParams).then(data => {router.push({ path: '/member/pay', query: { id: data.result.id } })})}return { checkoutInfo, changeAddress, submitOrder }

http://www.ppmy.cn/news/39404.html

相关文章

ChatGPT中文免登陆-ChatGPT中文版上线

ChatGPT不支持地区 ChatGPT 是一个开源平台&#xff0c;可在全球范围内使用&#xff0c;不应该存在地区限制。然而&#xff0c;由于某些原因&#xff0c;可能有地区对 ChatGPT 的访问有限制或屏蔽的情况。 如果您发现自己无法访问 ChatGPT&#xff0c;可以尝试以下解决方法&a…

ChatGPT的失败,是认知的失败,也是理解的失败

当下正在发生的事情&#xff0c;我们早已经历过&#xff0c;而且不止一次。当下以及可见的未来&#xff0c;ChatGPT除了“比人类更有效地”完成报告和写作等“智能”任务外&#xff0c;更多的恐怕是更多的“想象”了&#xff01;现在不是一些未知的东西正在做我们不知道的事情&…

Linux常用指令【文件目录操作】

linux 文件目录操作指令pwd 指令ls 指令cd 指令mkdir 指令rmdir 指令touch 指令cp 指令rm 指令mv 指令cat 指令more 指令less 指令> 和 >> 指令echo 指令head 指令tail 指令ln 指令history 指令pwd 指令 基本语法 pwd (显示当前工作目录的绝对路径) ls 指令 基本语法…

SQL——数据操作DML

对数据表中的数据的添加insert、删除delete、修改update操作。 目录 1 插入数据 2 删除数据 3 修改数据 1 插入数据 # 语法 insert into <tableName>(columnName,columnName....) values(value1,value2....);# 示例 ## 向数据表中指定的列添加数据&#xff08;不允许…

【视频文稿】车载Android应用开发与分析 - 开发系统应用

本期视频地址&#xff1a;https://www.bilibili.com/video/BV1NY411z7TK/ 前言 Hello&#xff0c;大家好&#xff0c;我是林栩。 开发车载应用&#xff0c;其实主要都是在Android系统中编写各种系统应用&#xff0c;所以上期视频先介绍了Android系统源码的下载和编译流程&…

算法学习day48

算法学习day481.力扣 198.打家劫舍1.1 分析1.2 代码2.力扣213.打家劫舍II2.1 分析2.2 代码3.力扣337.打家劫舍III3.1 分析3.2 代码4.参考资料1.力扣 198.打家劫舍 1.1 分析 1.确定dp数组以及下标的含义 dp[i] : 考虑下标i(包括i)以内的房屋&#xff0c;最多可以偷窃的金额数…

Java线程安全与等待通知

目录1. 线程不安全原因1.1 引入——线程不安全的例子&#xff08;抢占式执行&#xff09;1.2 线程不安全的原因&#xff08;5点其他&#xff09;2. 抢占式执行引起的线程不安全——synchronized3. 内存可见性引起的线程不安全——volatile3.1 例子——编译器优化误判3.2 volati…

点云学习(1): 获取点云的包络框

1. 记录一些容易忘记的点云操作----后续一定补充 1.获取点云的包络框 下面的get_axis_aligned_bounding_box(),get_min_bound(),get_max_bound()等函数非常好用 import open3d as o3d import numpy as np# 读取点云数据 pcd o3d.io.read_point_cloud("input.pcd"…