init
This commit is contained in:
247
project-wl-yonghuduan-uniapp-vue3/pages/address/index.scss
Normal file
247
project-wl-yonghuduan-uniapp-vue3/pages/address/index.scss
Normal file
@@ -0,0 +1,247 @@
|
||||
.address{
|
||||
background-color:#F3F5F9 ;
|
||||
height: 100vh;
|
||||
}
|
||||
.seachBox{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
background-color: white;
|
||||
::v-deep .uni-searchbar{
|
||||
padding-left: 40rpx;
|
||||
padding-right: 40rpx;
|
||||
|
||||
.uni-searchbar__box{
|
||||
border-radius: 34rpx!important;
|
||||
justify-content: flex-start;
|
||||
}
|
||||
}
|
||||
uni-search-bar{
|
||||
flex: auto;
|
||||
}
|
||||
.search-icon{
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url();
|
||||
}
|
||||
.search-btn{
|
||||
font-weight: bold;
|
||||
font-size: 28rpx;
|
||||
}
|
||||
}
|
||||
.list-container{
|
||||
padding: 0 18rpx;
|
||||
.checkbox{
|
||||
margin-right: 50rpx;
|
||||
}
|
||||
.address-list.isChecked{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
background-color:#F3F5F9 ;
|
||||
padding: 36rpx 0;
|
||||
padding-bottom: 0;
|
||||
padding-top: 0;
|
||||
}
|
||||
.address-item.isChecked{
|
||||
border-radius: 20rpx;
|
||||
padding: 36rpx 30rpx 0rpx;
|
||||
background: #FFFFFF;
|
||||
.address-item-footer{
|
||||
margin-bottom: 36rpx;
|
||||
}
|
||||
}
|
||||
.address-list{
|
||||
margin-top: 20rpx;
|
||||
background: #FFFFFF;
|
||||
border-radius: 20rpx;
|
||||
padding: 36rpx 30rpx 0rpx;
|
||||
padding-bottom: 30rpx;
|
||||
.address-item{
|
||||
flex: auto;
|
||||
}
|
||||
.address-name,.address-phone{
|
||||
display: inline-block;
|
||||
}
|
||||
.address-name{
|
||||
font-size: 30rpx;
|
||||
font-weight: bold;
|
||||
margin-right: 12rpx;
|
||||
}
|
||||
.address-phone,.address-address{
|
||||
color: #888;
|
||||
font-size: 26rpx;
|
||||
}
|
||||
.address-address{
|
||||
margin-top: 10rpx;
|
||||
}
|
||||
.line{
|
||||
margin-top: 20rpx;
|
||||
height: 2rpx;
|
||||
background-color:#F4F4F4 ;
|
||||
width: 100%;
|
||||
}
|
||||
.address-item-footer{
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
margin-top: 20rpx;
|
||||
.save-address{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
.label{
|
||||
font-size: 26rpx;
|
||||
color: #888;
|
||||
}
|
||||
.active,.checkbox{
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
margin-right: 10rpx;
|
||||
}
|
||||
.checkbox{
|
||||
background-image: url('../../static/checkbox.png');
|
||||
}
|
||||
.active{
|
||||
|
||||
background-image: url('../../static/checkboxActive.png');
|
||||
}
|
||||
}
|
||||
.right-box{
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.edit-btn,.delete-btn{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
icon{
|
||||
background: url(../../static/btn-bianji.png) no-repeat;
|
||||
background-size: contain;
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
margin-right: 6rpx;
|
||||
}
|
||||
text{
|
||||
font-size: 26rpx;
|
||||
color: #888;
|
||||
}
|
||||
}
|
||||
.delete-btn{
|
||||
margin-left: 20rpx;
|
||||
icon{
|
||||
background: url(../../static/btn-shanchu.png) no-repeat;
|
||||
background-size: contain;
|
||||
width: 30rpx;
|
||||
height: 30rpx;
|
||||
}
|
||||
}
|
||||
.right-box.active{
|
||||
.edit-btn,.delete-btn{
|
||||
text{
|
||||
color: #DADADA;
|
||||
}
|
||||
|
||||
}
|
||||
.edit-btn{
|
||||
icon{
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
background: url(../../static/btn-bianji-gray.png) no-repeat;
|
||||
background-size: contain;
|
||||
}
|
||||
|
||||
}
|
||||
.delete-btn{
|
||||
icon{
|
||||
width: 30rpx;
|
||||
height: 30rpx;
|
||||
background: url(../../static/btn-shanchu-gray.png) no-repeat;
|
||||
background-size: contain;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
.address-item-footer.active{
|
||||
justify-content: flex-end;
|
||||
}
|
||||
}
|
||||
|
||||
.address-footer{
|
||||
height:160rpx ;
|
||||
width: 100%;
|
||||
position: fixed;
|
||||
left: 0;
|
||||
bottom: 0;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
background-color: #FFFFFF;
|
||||
.manage-btn{
|
||||
width: 250rpx;
|
||||
height: 88rpx;
|
||||
border: 2rpx solid #979797;
|
||||
border-radius: 44rpx;
|
||||
color:#0F0F0F ;
|
||||
text-align: center;
|
||||
line-height: 88rpx;
|
||||
}
|
||||
.addAddress-btn{
|
||||
width: 400rpx;
|
||||
height: 88rpx;
|
||||
background-color:#CCCCCC ;
|
||||
border-radius: 44rpx;
|
||||
color:#FFFFFF ;
|
||||
text-align: center;
|
||||
line-height: 88rpx;
|
||||
margin-left: 26rpx;
|
||||
}
|
||||
.addAddress-btn.active{
|
||||
background-color:#E84134 ;
|
||||
}
|
||||
}
|
||||
|
||||
.all-select{
|
||||
display: flex;
|
||||
width: 100%;
|
||||
position: fixed;
|
||||
align-items: center;
|
||||
left: 0;
|
||||
bottom:160rpx ;
|
||||
height: 84rpx;
|
||||
background-color: #FFFFFF;
|
||||
border-bottom: 1px solid #F4F4F4;
|
||||
font-size: 24rpx;
|
||||
.checkbox{
|
||||
margin-right: 0rpx;
|
||||
margin-left:28rpx ;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
.empty-box{
|
||||
margin-top: 300rpx;
|
||||
image{
|
||||
width: 400rpx;
|
||||
height: 240rpx;
|
||||
display: block;
|
||||
margin: 0 auto;
|
||||
}
|
||||
view{
|
||||
font-size: 28rpx;
|
||||
color:#818181 ;
|
||||
margin-top: 40rpx;
|
||||
text-align: center;
|
||||
}
|
||||
}
|
||||
|
||||
.address-popup{
|
||||
::v-deep .uni-button-color{
|
||||
color: #E84134;
|
||||
}
|
||||
::v-deep .uni-dialog-title{
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
|
363
project-wl-yonghuduan-uniapp-vue3/pages/address/index.vue
Normal file
363
project-wl-yonghuduan-uniapp-vue3/pages/address/index.vue
Normal file
@@ -0,0 +1,363 @@
|
||||
<template>
|
||||
<nav-bar title="地址簿" :handleToLink="handleToLink"></nav-bar>
|
||||
<view class="address" v-if="netStatus">
|
||||
<view class="seachBox">
|
||||
<uni-search-bar cancelButton="none" placeholder="请输入姓名/手机号" v-model="searchValue" @input="input"
|
||||
@cancel="cancel" @clear="clear">
|
||||
<template v-slot:searchIcon>
|
||||
<view class="search-icon"></view>
|
||||
</template>
|
||||
</uni-search-bar>
|
||||
</view>
|
||||
<view class="list-container">
|
||||
<scroll-view v-if="list.data.length > 0" scroll-y :style="scrollHeight" lower-threshold="30"
|
||||
@scrolltolower="LoadMoreCustomers">
|
||||
<!-- 列表内容 -->
|
||||
<view v-for="(item, index) in list.data" @click.stop.prevent="
|
||||
isManage ? checkbox(index, item) : handleToAddressInfo(item.id)
|
||||
" :key="index" class="address-list" :class="isManage ? 'isChecked' : ''">
|
||||
<view class="checkbox" v-if="isManage">
|
||||
<view class="checkRadio">
|
||||
<radio style="transform: scale(0.77)" color="#F23C24" :value="String(index)"
|
||||
:checked="item.selected" />
|
||||
</view>
|
||||
</view>
|
||||
<view class="address-item" :class="isManage ? 'isChecked' : ''">
|
||||
<view class="address-name">{{ item.name }}</view>
|
||||
<view class="address-phone">{{ item.phoneNumber }}</view>
|
||||
<view class="address-address">
|
||||
{{
|
||||
item.province
|
||||
? item.province.name +
|
||||
item.city.name +
|
||||
item.county.name +
|
||||
item.address
|
||||
: ''
|
||||
}}
|
||||
</view>
|
||||
<view class="line"></view>
|
||||
<view class="address-item-footer" :class="type === 'get' ? 'active' : ''">
|
||||
<view class="save-address" v-if="type !== 'get'" @click.stop="
|
||||
handleSaveToDefaultAddress(item.id, item.isDefault)
|
||||
">
|
||||
<view class="checkbox" :class="{ active: Boolean(item.isDefault) }"></view>
|
||||
<view class="label">默认寄件地址</view>
|
||||
</view>
|
||||
<view class="right-box" :class="!isManage ? '' : 'active'">
|
||||
<view class="edit-btn" @click.stop="handleEditAddress(item)">
|
||||
<icon></icon>
|
||||
<text>编辑</text>
|
||||
</view>
|
||||
<view class="delete-btn" @click.stop="handledDelete(item.id)">
|
||||
<icon></icon>
|
||||
<text>删除</text>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 加载底部 -->
|
||||
<uni-load-more :status="status" />
|
||||
</scroll-view>
|
||||
<!-- 空页面 -->
|
||||
<view v-else class="empty-box">
|
||||
<image src="../../static/address-empty.png"></image>
|
||||
<view>{{ searchValue ? '没有搜索到相关条件的地址' : '暂无数据' }}</view>
|
||||
</view>
|
||||
<!-- 全选操作栏 -->
|
||||
<view class="all-select" v-if="isManage">
|
||||
<view class="checkbox" @click="selectAll">
|
||||
<view class="checkRadio">
|
||||
<radio style="transform: scale(0.77)" :checked="isAllSelect" color="#F23C24" />
|
||||
</view>
|
||||
</view>
|
||||
<view>全选</view>
|
||||
</view>
|
||||
|
||||
<!-- 底部操作栏 -->
|
||||
<view class="address-footer">
|
||||
<view class="manage-btn" v-if="list.data.length > 0" @click="handleEdit">{{ isManage ? '完成' : '管理' }}
|
||||
</view>
|
||||
<view class="addAddress-btn" :class="
|
||||
(isManage && deleteIds.data.length) || !isManage ? 'active' : ''
|
||||
" @click="handleDeleteOrAdd">{{ isManage ? '删除' : '新增地址' }}</view>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 断网显示的页面 -->
|
||||
<net-fail v-else :handleToRefresh="handleToRefresh"></net-fail>
|
||||
|
||||
<!-- 删除确认对话框 -->
|
||||
<uni-popup ref="popup" type="dialog" class="address-popup">
|
||||
<uni-popup-dialog mode="base" :content="isManage ? '确定是否删除所选地址?' : '确定是否删除此条地址?'" :animation="false"
|
||||
:before-close="true" @close="close" @confirm="confirm">
|
||||
</uni-popup-dialog>
|
||||
</uni-popup>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
reactive,
|
||||
onMounted,
|
||||
} from 'vue'
|
||||
import {
|
||||
onLoad,
|
||||
onPullDownRefresh
|
||||
} from '@dcloudio/uni-app'
|
||||
import {
|
||||
getAddressList,
|
||||
deleteAddress,
|
||||
editAddress
|
||||
} from '@/pages/api/address.js'
|
||||
let scrollHeight = ref('')
|
||||
let searchValue = ref('')
|
||||
let popup = ref(null)
|
||||
let isAllSelect = ref(false)
|
||||
let isManage = ref(false)
|
||||
let deleteIds = reactive({
|
||||
data: [] //删除的地址id集合
|
||||
})
|
||||
let list = reactive({
|
||||
data: [] //列表数据
|
||||
})
|
||||
let pageInfo = reactive({
|
||||
page: 1,
|
||||
pageSize: 10
|
||||
})
|
||||
let status = ref('more') //加载状态
|
||||
let type = ref('')
|
||||
let isFromAddress = ref('')
|
||||
let netStatus = ref(true)
|
||||
// 监听页面加载执行
|
||||
onLoad((options) => {
|
||||
type.value = options.type
|
||||
isFromAddress.value = options.isFromAddress
|
||||
uni.getSystemInfo({
|
||||
success: (res) => {
|
||||
scrollHeight.value = 'height:' + (res.screenHeight - 190) + 'px'
|
||||
}
|
||||
})
|
||||
})
|
||||
onMounted(() => {
|
||||
getList()
|
||||
})
|
||||
// 监听下拉刷新
|
||||
onPullDownRefresh(() => {
|
||||
pageInfo.page = 1
|
||||
getList()
|
||||
})
|
||||
// 断网刷新
|
||||
const handleToRefresh = () => {
|
||||
pageInfo.page = 1
|
||||
getList()
|
||||
}
|
||||
const input = (e) => {
|
||||
searchValue.value = e
|
||||
pageInfo.page = 1
|
||||
getList()
|
||||
}
|
||||
const clear = () => {
|
||||
|
||||
}
|
||||
//设置是否为默认寄件地址
|
||||
const handleSaveToDefaultAddress = (id, isDefault) => {
|
||||
if (isManage.value) {
|
||||
return
|
||||
}
|
||||
// 修改默认地址
|
||||
editAddress({
|
||||
id,
|
||||
isDefault: isDefault === 1 ? 0 : 1
|
||||
}).then((res) => {
|
||||
pageInfo.page = 1
|
||||
getList()
|
||||
if (res.code === 200) {
|
||||
uni.showToast({
|
||||
title: '修改成功',
|
||||
icon: 'none',
|
||||
duration: 1000,
|
||||
type: 'success'
|
||||
})
|
||||
} else {
|
||||
uni.showToast({
|
||||
title: '修改成功',
|
||||
icon: 'none',
|
||||
duration: 1000, //提示的延迟时间,单位毫秒,默认:1500
|
||||
type: 'error'
|
||||
})
|
||||
}
|
||||
|
||||
}).catch(() => {
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
})
|
||||
})
|
||||
}
|
||||
//返回上一级防止是最后一级报错
|
||||
const handleToLink = () => {
|
||||
if (type.value !== 'address') {
|
||||
uni.navigateBack()
|
||||
} else {
|
||||
uni.switchTab({
|
||||
url: '/pages/my/index'
|
||||
})
|
||||
}
|
||||
|
||||
}
|
||||
//跳转回寄件收件选择地址
|
||||
const handleToAddressInfo = (id) => {
|
||||
if (type.value === 'address') {
|
||||
return
|
||||
} else if (type.value === 'get') {
|
||||
if (uni.getStorageSync('sendId') === id) return uni.showToast({
|
||||
title: '寄件地址和收件地址不能选择同一个',
|
||||
icon: 'none',
|
||||
duration: 1000
|
||||
})
|
||||
} else {
|
||||
if (uni.getStorageSync('getId') === id) return uni.showToast({
|
||||
title: '寄件地址和收件地址不能选择同一个',
|
||||
icon: 'none',
|
||||
duration: 1000
|
||||
})
|
||||
}
|
||||
// 关闭当前页面,跳转回寄件收件选择地址
|
||||
uni.redirectTo({
|
||||
url: '/pages/express-delivery/index?type=' + type.value + '&editOrAdd=edit&id=' + id +
|
||||
'&isFromAddress=true'
|
||||
})
|
||||
}
|
||||
//批量删除或者新增地址
|
||||
const handleDeleteOrAdd = () => {
|
||||
if (isManage.value) {
|
||||
if (deleteIds.data.length) popup.value.open()
|
||||
} else {
|
||||
uni.redirectTo({
|
||||
url: '/subPages/address-info/index?isFromAddress=' + isFromAddress.value +
|
||||
'&editOrAdd=add&type=' + type.value
|
||||
})
|
||||
}
|
||||
}
|
||||
//编辑地址
|
||||
const handleEditAddress = (item) => {
|
||||
if (isManage.value) {
|
||||
return
|
||||
}
|
||||
uni.redirectTo({
|
||||
url: '/subPages/address-info/index?id=' + item.id + '&isFromAddress=' + isFromAddress.value +
|
||||
'&editOrAdd=edit' +
|
||||
'&type=' + type.value + '&isDefault=' + item.isDefault
|
||||
})
|
||||
}
|
||||
//获取列表数据
|
||||
const getList = (flag) => {
|
||||
status.value = 'loading'
|
||||
getAddressList({
|
||||
page: pageInfo.page,
|
||||
pageSize: pageInfo.pageSize,
|
||||
keyword: searchValue.value
|
||||
}).then((res) => {
|
||||
let arr = res.data.items ? res.data.items : []
|
||||
status.value = arr.length < 10 ? 'no-more' : 'more'
|
||||
//用于区分是一次进页面记载数据还是后续上拉加载
|
||||
if (flag === 'topPull') {
|
||||
list.data = list.data.concat(arr)
|
||||
} else {
|
||||
list.data = arr
|
||||
}
|
||||
// 与搜索框等内容关联全选矿和具体选中状态,如果list.data中所有的selected都为true,即全选,全选文字前的radio框高亮
|
||||
if (list.data.every((item) => item.selected)) {
|
||||
isAllSelect.value = true
|
||||
} else {
|
||||
isAllSelect.value = false
|
||||
}
|
||||
uni.stopPullDownRefresh()
|
||||
netStatus.value = true
|
||||
console.log(456)
|
||||
}).catch((err) => {
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
})
|
||||
netStatus.value = false
|
||||
console.log(123)
|
||||
})
|
||||
}
|
||||
//上拉加载
|
||||
const LoadMoreCustomers = () => {
|
||||
pageInfo.page = pageInfo.page + 1
|
||||
if (status.value === 'no-more') {
|
||||
return
|
||||
}
|
||||
getList('topPull')
|
||||
}
|
||||
//管理地址簿
|
||||
const handleEdit = () => {
|
||||
isManage.value = !isManage.value
|
||||
}
|
||||
|
||||
//删除
|
||||
const handledDelete = (id) => {
|
||||
if (isManage.value) {
|
||||
return
|
||||
}
|
||||
popup.value.open()
|
||||
deleteIds.data.push(id)
|
||||
}
|
||||
|
||||
//关闭删除确认提示框
|
||||
const close = () => {
|
||||
popup.value.close()
|
||||
deleteIds.data = []
|
||||
}
|
||||
|
||||
//确认删除
|
||||
const confirm = () => {
|
||||
popup.value.close()
|
||||
deleteAddress(deleteIds.data).then((res) => {
|
||||
pageInfo.page = 1
|
||||
pageInfo.pageSize = 10
|
||||
getList()
|
||||
uni.showToast({
|
||||
title: '删除成功',
|
||||
icon: 'success',
|
||||
duration: 1000
|
||||
})
|
||||
isManage.value = false
|
||||
}).catch(() => {
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
//选中,与取消选中(和全选联动)
|
||||
const checkbox = (index, item) => {
|
||||
list.data[index].selected = !list.data[index].selected
|
||||
// 如果list.data中所有的selected都为true,即全选,全选前的radio框高亮
|
||||
if (list.data.every((item) => item.selected)) {
|
||||
isAllSelect.value = true
|
||||
} else {
|
||||
isAllSelect.value = false
|
||||
}
|
||||
deleteIds.data = list.data.filter(item => item.selected).map(item => item.id)
|
||||
}
|
||||
//全选或者全不选
|
||||
const selectAll = () => {
|
||||
isAllSelect.value = !isAllSelect.value
|
||||
list.data = list.data.map((item) => {
|
||||
return Object.assign({}, item, {
|
||||
selected: isAllSelect.value
|
||||
})
|
||||
})
|
||||
deleteIds.data = list.data.filter(item => item.selected).map(item => item.id)
|
||||
}
|
||||
</script>
|
||||
|
||||
<style src="./index.scss" lang="scss" scoped></style>
|
50
project-wl-yonghuduan-uniapp-vue3/pages/api/address.js
Normal file
50
project-wl-yonghuduan-uniapp-vue3/pages/api/address.js
Normal file
@@ -0,0 +1,50 @@
|
||||
import {
|
||||
request
|
||||
} from "../../utils/request.js"
|
||||
|
||||
//地址簿列表
|
||||
export const getAddressList = (params) =>
|
||||
request({
|
||||
url: `/address/page`,
|
||||
method: 'get',
|
||||
params
|
||||
})
|
||||
|
||||
//查询地址详情
|
||||
export const getAddressInfoDetail = (params) =>
|
||||
request({
|
||||
url: `/address/detail/` + params,
|
||||
method: 'get',
|
||||
params
|
||||
})
|
||||
|
||||
//新建地址簿
|
||||
export const addAddress = (params) =>
|
||||
request({
|
||||
url: `/address`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
||||
|
||||
//修改地址簿
|
||||
export const editAddress = (params) =>
|
||||
request({
|
||||
url: `/address`,
|
||||
method: 'put',
|
||||
params
|
||||
})
|
||||
|
||||
//删除地址簿
|
||||
export const deleteAddress = (params) =>
|
||||
request({
|
||||
url: `/address`,
|
||||
method: 'delete',
|
||||
params
|
||||
})
|
||||
//查询用户默认寄件地址
|
||||
export const defaultAddress = (params) =>
|
||||
request({
|
||||
url: `/address/defaultAddress`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
11
project-wl-yonghuduan-uniapp-vue3/pages/api/common.js
Normal file
11
project-wl-yonghuduan-uniapp-vue3/pages/api/common.js
Normal file
@@ -0,0 +1,11 @@
|
||||
import {
|
||||
request
|
||||
} from "../../utils/request.js"
|
||||
|
||||
// 获取省市区
|
||||
export const getProvinces = (params) =>
|
||||
request({
|
||||
url: '/areas/children',
|
||||
method: 'get',
|
||||
params
|
||||
})
|
11
project-wl-yonghuduan-uniapp-vue3/pages/api/freight.js
Normal file
11
project-wl-yonghuduan-uniapp-vue3/pages/api/freight.js
Normal file
@@ -0,0 +1,11 @@
|
||||
import {
|
||||
request
|
||||
} from "../../utils/request.js"
|
||||
|
||||
// 计算运费
|
||||
export const calculateFreight = (params) =>
|
||||
request({
|
||||
url: '/tasks/calculate',
|
||||
method: 'post',
|
||||
params
|
||||
})
|
18
project-wl-yonghuduan-uniapp-vue3/pages/api/login.js
Normal file
18
project-wl-yonghuduan-uniapp-vue3/pages/api/login.js
Normal file
@@ -0,0 +1,18 @@
|
||||
import {
|
||||
request
|
||||
} from "../../utils/request.js"
|
||||
|
||||
// 手机号登录
|
||||
export const login = (params) =>
|
||||
request({
|
||||
url: `/user/login`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
||||
// 刷新token
|
||||
export const resetToken = (params) =>
|
||||
request({
|
||||
url: `/user/refresh`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
24
project-wl-yonghuduan-uniapp-vue3/pages/api/my.js
Normal file
24
project-wl-yonghuduan-uniapp-vue3/pages/api/my.js
Normal file
@@ -0,0 +1,24 @@
|
||||
import {
|
||||
request
|
||||
} from "../../utils/request.js"
|
||||
// 获取用户信息
|
||||
export const getUserInfo = (params) =>
|
||||
request({
|
||||
url: `/user/profile`,
|
||||
method: 'get',
|
||||
params
|
||||
})
|
||||
// 获取用户实名认证
|
||||
export const getRealNameStatusApi = (params) =>
|
||||
request({
|
||||
url: `/user/realNameVerify`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
||||
//修改用户信息
|
||||
export const updateUserInfo = (params) =>
|
||||
request({
|
||||
url: `/user/profile`,
|
||||
method: 'put',
|
||||
params
|
||||
})
|
88
project-wl-yonghuduan-uniapp-vue3/pages/api/order.js
Normal file
88
project-wl-yonghuduan-uniapp-vue3/pages/api/order.js
Normal file
@@ -0,0 +1,88 @@
|
||||
import {
|
||||
request
|
||||
} from "../../utils/request.js"
|
||||
// 省市区三级联动
|
||||
export const getArea = (params) =>
|
||||
request({
|
||||
url: `/areas/children`,
|
||||
method: 'get',
|
||||
params
|
||||
})
|
||||
|
||||
|
||||
//获取常用货物列表
|
||||
export const usualGoodsList = (params) =>
|
||||
request({
|
||||
url: `/order-manager/cargo/hot`,
|
||||
method: 'get',
|
||||
params
|
||||
})
|
||||
|
||||
//获取货物列表
|
||||
export const goodsList = (params) =>
|
||||
request({
|
||||
url: `/order-manager/cargo/last`,
|
||||
method: 'get',
|
||||
params
|
||||
})
|
||||
|
||||
//下单
|
||||
export const doOrder = (params) =>
|
||||
request({
|
||||
url: `/order-manager/order`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
||||
//获取订单详情
|
||||
export const getOrderDetail = (params) =>
|
||||
request({
|
||||
url: `/order-manager/order/` + params,
|
||||
method: 'get',
|
||||
})
|
||||
//获取订单列表
|
||||
export const getOrderList = (params) =>
|
||||
request({
|
||||
url: `/order-manager/order/page`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
||||
//获取预估总价
|
||||
export const getEstimatePrice = (params) =>
|
||||
request({
|
||||
url: `/order-manager/order/totalPrice`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
||||
//取消订单
|
||||
export const cancelOrder = (params) =>
|
||||
request({
|
||||
url: `/order-manager/order/cancel/` + params,
|
||||
method: 'put',
|
||||
params
|
||||
})
|
||||
//支付订单
|
||||
export const payOrder = (params) =>
|
||||
request({
|
||||
url: `/order-manager/order/pay`,
|
||||
method: 'put',
|
||||
params
|
||||
})
|
||||
//删除订单
|
||||
export const deleteOrder = (params) =>
|
||||
request({
|
||||
url: `/order-manager/order/del/` + params,
|
||||
method: 'put',
|
||||
params
|
||||
})
|
||||
//获取查快递的寄件和收件的数量
|
||||
export const getGoodsNum = (params) =>
|
||||
request({
|
||||
url: `/order-manager/order/count`,
|
||||
method: 'get',
|
||||
})
|
||||
//获取订单轨迹
|
||||
export const getOrderLine = (params) =>
|
||||
request({
|
||||
url: `/order-manager/order/track/` + params,
|
||||
method: 'get',
|
||||
})
|
32
project-wl-yonghuduan-uniapp-vue3/pages/api/user.js
Normal file
32
project-wl-yonghuduan-uniapp-vue3/pages/api/user.js
Normal file
@@ -0,0 +1,32 @@
|
||||
import {
|
||||
request
|
||||
} from "../../utils/request.js"
|
||||
|
||||
// 手机号登录
|
||||
export const phoneLogins = (params) =>
|
||||
request({
|
||||
url: `/logins/phone`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
||||
// 账号登录
|
||||
export const userLogins = (params) =>
|
||||
request({
|
||||
url: `/login/account`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
||||
// 发送短信验证码
|
||||
export const getsmsCode = (params) =>
|
||||
request({
|
||||
url: `/verifyCodes/smsCode`,
|
||||
method: 'post',
|
||||
params
|
||||
})
|
||||
// 获取用户信息
|
||||
export const getUserInfo = (params) =>
|
||||
request({
|
||||
url: `/users/get`,
|
||||
method: 'get',
|
||||
params
|
||||
})
|
@@ -0,0 +1,77 @@
|
||||
.getTimePicker {
|
||||
::v-deep .uni-popup__wrapper {
|
||||
height: 680rpx;
|
||||
background-color: white !important;
|
||||
border-radius: 24rpx 24rpx 0 0;
|
||||
}
|
||||
|
||||
.header {
|
||||
display: flex;
|
||||
height: 56rpx;
|
||||
padding: 30rpx 38rpx;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
border-radius: 24rpx 24rpx 0 0;
|
||||
|
||||
.header-title {
|
||||
font-size: 32rpx;
|
||||
color: #151515;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.close {
|
||||
width: 24rpx;
|
||||
height: 24rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url('../../../static/icon21.png');
|
||||
}
|
||||
}
|
||||
.time-select{
|
||||
display: flex;
|
||||
.select-day{
|
||||
width: 296rpx;
|
||||
height: 500rpx;
|
||||
background: #F6F6F6;
|
||||
.select-day-item{
|
||||
height: 108rpx;
|
||||
line-height: 108rpx;
|
||||
text-align: center;
|
||||
font-size: 28rpx;
|
||||
}
|
||||
.select-day-item.active{
|
||||
background: #fff;
|
||||
color:#E84134 ;
|
||||
}
|
||||
}
|
||||
.select-time{
|
||||
.time-item{
|
||||
font-size: 28rpx;
|
||||
display: flex;
|
||||
padding-left: 80rpx;
|
||||
margin-bottom: 46rpx;
|
||||
.time-value{
|
||||
padding-right: 126rpx;
|
||||
}
|
||||
.time-select{
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url('../../../static/gouxuan.png');
|
||||
}
|
||||
}
|
||||
.time-item.active{
|
||||
color:#E84134 ;
|
||||
}
|
||||
::v-deep scroll-view{
|
||||
height:500rpx ;
|
||||
}
|
||||
|
||||
//隐藏滚动条
|
||||
::v-deep ::-webkit-scrollbar {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,103 @@
|
||||
<!-- 期待上门时间弹窗 -->
|
||||
<template>
|
||||
<view class="getTimePicker">
|
||||
<uni-popup ref="popup" type="bottom" :safe-area="false">
|
||||
<view class="header">
|
||||
<view class="header-title">期望上门时间</view>
|
||||
<view class="close" @click="handleCancel"></view>
|
||||
</view>
|
||||
<!-- 时间选择区域 -->
|
||||
<view class="time-select">
|
||||
<view class="select-day">
|
||||
<view :class="selectedDay === index ?'active':''" class="select-day-item"
|
||||
@click="handleSelectDay(index,item)" v-for="(item,index) in selectDay" :key='index'>
|
||||
{{item}}
|
||||
</view>
|
||||
</view>
|
||||
<view class="select-time">
|
||||
<scroll-view scroll-y :scroll-top="scrollTop" @scroll="scroll">
|
||||
<view :class="selectedTime === item.value ?'active':''" class="time-item"
|
||||
v-for="(item,index) in (selectedDay===0?todayList.todos:dateList)" :key="index"
|
||||
@click='handleSelectTime(item.value,item.label)'>
|
||||
<view class="time-value">{{item.label}}</view>
|
||||
<view class="time-select" v-if='selectedTime === item.value'></view>
|
||||
</view>
|
||||
</scroll-view>
|
||||
</view>
|
||||
</view>
|
||||
</uni-popup>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
reactive,
|
||||
onMounted,
|
||||
} from 'vue';
|
||||
const emits = defineEmits(["@getTime"]);
|
||||
const popup = ref()
|
||||
let scrollTop = ref(0) //顶部位置
|
||||
let selectedDay = ref(0)
|
||||
let selectedDayLabel = ref('今天')
|
||||
let selectedTime = ref(0)
|
||||
let selectedTimeLabel = ref()
|
||||
const selectDay = reactive(['今天', '明天', '后天'])
|
||||
let todayList = reactive({
|
||||
todos: [{
|
||||
label: '一小时内',
|
||||
value: 1
|
||||
}]
|
||||
})
|
||||
onMounted(() => {
|
||||
todayList.todos = [...todayList.todos.concat(
|
||||
dateList.filter(item => item.value > new Date().getHours())
|
||||
)]
|
||||
})
|
||||
const dateList = reactive(
|
||||
// 弹窗中的时间列表
|
||||
Array.from({
|
||||
length: 11
|
||||
}, (v, k) => ({
|
||||
label: `${k + 9}:00-${k + 10}:00`,
|
||||
value: k + 9
|
||||
}))
|
||||
)
|
||||
const scroll = (e) => {
|
||||
scrollTop.value = e.detail.scrollTop
|
||||
}
|
||||
//选择具体时间段
|
||||
const handleSelectTime = (index, item) => {
|
||||
selectedTime.value = index
|
||||
selectedTimeLabel.value = item
|
||||
popup.value.close('bottom');
|
||||
emits('getTime', {
|
||||
selectedDay: selectedDay.value,
|
||||
selectedDayLabel: selectedDayLabel.value,
|
||||
selectedTime: selectedTime.value,
|
||||
selectedTimeLabel: selectedTimeLabel.value
|
||||
})
|
||||
}
|
||||
//选择哪天
|
||||
const handleSelectDay = (index, item) => {
|
||||
selectedDay.value = index
|
||||
scrollTop.value = 0
|
||||
selectedTime.value = 0
|
||||
selectedDayLabel.value = item
|
||||
}
|
||||
// 打开弹层
|
||||
const handleOpen = () => {
|
||||
popup.value.open('bottom');
|
||||
|
||||
};
|
||||
// 关闭弹层
|
||||
const handleCancel = () => {
|
||||
popup.value.close('bottom');
|
||||
}
|
||||
// 暴漏给父组件
|
||||
defineExpose({
|
||||
handleOpen
|
||||
});
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped src='./getTimePicker.scss'></style>
|
@@ -0,0 +1,80 @@
|
||||
.payType{
|
||||
::v-deep .uni-popup__wrapper {
|
||||
height:480rpx;
|
||||
background-color: white !important;
|
||||
border-radius: 24rpx 24rpx 0 0;
|
||||
padding: 29rpx 38rpx 43rpx;
|
||||
}
|
||||
|
||||
.headers {
|
||||
display: flex;
|
||||
height: 56rpx;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
border-radius: 24rpx 24rpx 0 0;
|
||||
|
||||
.header-title {
|
||||
font-size: 32rpx;
|
||||
color: #151515;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.close {
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url('../../../static/guanbi.png');
|
||||
}
|
||||
}
|
||||
.contents{
|
||||
margin-top: 64rpx;
|
||||
.content-item{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
.left-content{
|
||||
.title{
|
||||
font-size: 28rpx;
|
||||
color: #151515;
|
||||
margin-bottom: 20rpx;
|
||||
}
|
||||
.subTitle{
|
||||
font-size: 24rpx;
|
||||
color: #888888;
|
||||
}
|
||||
}
|
||||
.right-content{
|
||||
.active,.checkbox{
|
||||
width: 50rpx;
|
||||
height: 50rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
}
|
||||
.checkbox{
|
||||
background-image: url('../../../static/checkbox.png');
|
||||
}
|
||||
.active{
|
||||
|
||||
background-image: url('../../../static/checkboxActive.png');
|
||||
}
|
||||
}
|
||||
}
|
||||
.content-item:last-child{
|
||||
margin-top: 60rpx;
|
||||
}
|
||||
}
|
||||
.footers{
|
||||
.btn{
|
||||
width: 404rpx;
|
||||
height: 88rpx;
|
||||
background-color: #E84134;
|
||||
border-radius: 44rpx;
|
||||
text-align: center;
|
||||
line-height: 88rpx;
|
||||
color: white;
|
||||
margin: 0 auto;
|
||||
margin-top: 43rpx;
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,70 @@
|
||||
<!-- 付款方式弹窗 -->
|
||||
<template>
|
||||
<view class="payType">
|
||||
<uni-popup ref="popup" type="bottom" :safe-area="false">
|
||||
<view class="headers">
|
||||
<view class="header-title">付款方式</view>
|
||||
<view class="close" @click="handleCancel"></view>
|
||||
</view>
|
||||
<view class="contents">
|
||||
<view class="content-item" v-for="(item,index) in list" :key="index" @click="handleChangePayType(item)">
|
||||
<view class="left-content">
|
||||
<view class="title">{{item.title}}</view>
|
||||
<view class="subTitle">{{item.subTitle}}</view>
|
||||
</view>
|
||||
<view class="right-content">
|
||||
<view class="checkbox" :class="{active:item.value===payType}"></view>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<view class="footers" @click="handleCancel">
|
||||
<view class="btn">确定</view>
|
||||
</view>
|
||||
</uni-popup>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
reactive,
|
||||
} from 'vue';
|
||||
const emits = defineEmits(["@getPayType"]);
|
||||
const popup = ref()
|
||||
const payType = ref(1)
|
||||
const list = reactive([
|
||||
{
|
||||
title:'寄付',
|
||||
subTitle:'快递员取件时,寄方可在线支付、扫快递员收款码进行支付',
|
||||
value:1
|
||||
},
|
||||
{
|
||||
title:'到付',
|
||||
subTitle:'快递签收后,收方可通过扫快递员收款码进行支付',
|
||||
value:2
|
||||
}
|
||||
])
|
||||
//选择付款方式
|
||||
const handleChangePayType =(item)=>{
|
||||
payType.value = item.value
|
||||
}
|
||||
// 打开弹层
|
||||
const handleOpen = () => {
|
||||
popup.value.open('bottom');
|
||||
|
||||
};
|
||||
// 关闭弹层
|
||||
const handleCancel = () => {
|
||||
popup.value.close('bottom');
|
||||
emits('getPayType',payType.value)
|
||||
}
|
||||
// 暴漏给父组件
|
||||
defineExpose({
|
||||
handleOpen
|
||||
});
|
||||
</script>
|
||||
|
||||
<style lang="scss" src='./payType.scss' scoped></style>
|
||||
|
||||
|
||||
|
@@ -0,0 +1,351 @@
|
||||
.express-delivery{
|
||||
background-color: #F3F5F9 !important;
|
||||
padding: 20rpx;
|
||||
height: 80vh;
|
||||
.address-box{
|
||||
background: #FFFFFF;
|
||||
border-radius: 20rpx;
|
||||
display: flex;
|
||||
padding: 42rpx 28rpx;
|
||||
padding-bottom: 0rpx;
|
||||
.left{
|
||||
margin-right: 20rpx;
|
||||
.send,.get{
|
||||
width: 40rpx;
|
||||
height: 40prx;
|
||||
border-radius: 50%;
|
||||
text-align: center;
|
||||
line-height: 40rpx;
|
||||
font-size: 24rpx;
|
||||
}
|
||||
.send{
|
||||
background: #000000;
|
||||
color: white;
|
||||
}
|
||||
.get{
|
||||
background: #E63E32;
|
||||
color: white;
|
||||
}
|
||||
.line{
|
||||
width: 2rpx;
|
||||
height: 120rpx;
|
||||
border-left: 2rpx dashed #D5D1D1;
|
||||
margin-left: 18rpx;
|
||||
}
|
||||
.active{
|
||||
height: 153rpx;
|
||||
}
|
||||
}
|
||||
.right{
|
||||
width: 100%;
|
||||
position: relative;
|
||||
.send-people,.get-people{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
.send-title{
|
||||
max-width: 490rpx;
|
||||
height: 100%;
|
||||
}
|
||||
.send-sub-title{
|
||||
color: #151515;
|
||||
font-size: 32rpx;
|
||||
margin-bottom: 6rpx;
|
||||
font-weight: bold;
|
||||
text{
|
||||
color: #888888;
|
||||
font-size: 24rpx;
|
||||
}
|
||||
}
|
||||
.send-desc{
|
||||
color: #888888;
|
||||
font-size: 24rpx;
|
||||
}
|
||||
.address-enter{
|
||||
font-weight: bold;
|
||||
border-left: 2rpx solid #F4F4F4;
|
||||
padding-left: 28rpx;
|
||||
font-size: 24rpx;
|
||||
}
|
||||
}
|
||||
.get-people{
|
||||
margin-bottom: 40rpx;
|
||||
}
|
||||
.send-people{
|
||||
margin-bottom: 40rpx;
|
||||
}
|
||||
.line{
|
||||
width: 100%;
|
||||
height: 2rpx;
|
||||
background-color: #F4F4F4;
|
||||
margin-bottom: 40rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.send-form{
|
||||
height: 400rpx;
|
||||
width: 100%;
|
||||
border-radius: 20rpx;
|
||||
margin-top: 40rpx;
|
||||
|
||||
.tab-box{
|
||||
display: flex;
|
||||
position: relative;
|
||||
background-color: white;
|
||||
border-radius: 0 20rpx 0 0;
|
||||
.tab-item{
|
||||
flex: 1;
|
||||
text-align: center;
|
||||
height: 76rpx;
|
||||
line-height: 76rpx;
|
||||
font-size: 30rpx;
|
||||
background-color:#E3E7ED ;
|
||||
border-radius:0 20rpx 0 20rpx ;
|
||||
position: relative;
|
||||
}
|
||||
.tab-item.active{
|
||||
font-weight: bold;
|
||||
height: 96rpx;
|
||||
line-height: 96rpx;
|
||||
background-color:white ;
|
||||
border-radius:20rpx 20rpx 0 0rpx ;
|
||||
position: relative;
|
||||
bottom: 20rpx;
|
||||
}
|
||||
.isNotActive.tab-item:first-child{
|
||||
border-radius:20rpx 0rpx 20rpx 0rpx ;
|
||||
}
|
||||
}
|
||||
.tab-box.active{
|
||||
border-radius: 20rpx 0 0 0;
|
||||
}
|
||||
.form-box{
|
||||
background-color: #FFFFFF;
|
||||
position: relative;
|
||||
bottom: 20rpx;
|
||||
padding: 0 36rpx 0 28rpx;
|
||||
border-radius: 0 0 20rpx 20rpx;
|
||||
.form-getTime,.goods-info,.pay-type{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
height: 96rpx;
|
||||
border-bottom: 2rpx solid #F4F4F4;
|
||||
.left{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
.label{
|
||||
font-size: 26rpx;
|
||||
color:#151515 ;
|
||||
font-weight: bold;
|
||||
}
|
||||
.required{
|
||||
margin-left: 12rpx;
|
||||
width: 62rpx;
|
||||
height: 30rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url('../../static/required.png');
|
||||
}
|
||||
}
|
||||
.right{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
.nextIcon{
|
||||
margin-left: 18rpx;
|
||||
}
|
||||
.value{
|
||||
color: #888888;
|
||||
font-size: 26rpx;
|
||||
}
|
||||
.value.active{
|
||||
color: #151515;
|
||||
font-weight: bold;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
.pay-type{
|
||||
border-bottom: none;
|
||||
}
|
||||
}
|
||||
}
|
||||
.footer{
|
||||
height:172rpx ;
|
||||
background-color: white;
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
position: fixed;
|
||||
width: 100%;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
padding-top: 22rpx;
|
||||
.right{
|
||||
.btn{
|
||||
width: 268rpx;
|
||||
height: 88rpx;
|
||||
background-color: #CCCCCC;
|
||||
border-radius: 44rpx;
|
||||
text-align: center;
|
||||
line-height: 88rpx;
|
||||
color: white;
|
||||
margin-right: 29rpx;
|
||||
}
|
||||
.btn.active{
|
||||
background-color:#E84134 ;
|
||||
}
|
||||
}
|
||||
.left{
|
||||
margin-left: 29rpx;
|
||||
.left-top{
|
||||
font-size:26rpx ;
|
||||
color:#323232 ;
|
||||
display: flex;
|
||||
.all-account{
|
||||
border-right: 2rpx solid #F4F4F4;
|
||||
display: flex;
|
||||
padding-right: 10rpx;
|
||||
margin-right: 10rpx;
|
||||
.price{
|
||||
color: #E73F33;
|
||||
}
|
||||
}
|
||||
.price-detail{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
.arrow{
|
||||
width: 0;
|
||||
height: 0;
|
||||
border: 10rpx solid transparent;
|
||||
border-top-color: black;
|
||||
margin-left: 10rpx;
|
||||
margin-top: 8rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
.left-bottom{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
margin-top: 22rpx;
|
||||
.active,.checkbox{
|
||||
width: 32rpx;
|
||||
height: 32rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
margin-right: 10rpx;
|
||||
}
|
||||
.checkbox{
|
||||
background-image: url('../../static/checkbox.png');
|
||||
}
|
||||
.active{
|
||||
|
||||
background-image: url('../../static/checkboxActive.png');
|
||||
}
|
||||
.content{
|
||||
color: #696969;
|
||||
font-size: 20rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
.footer.active{
|
||||
z-index: 9999;
|
||||
}
|
||||
.priceDetail{
|
||||
::v-deep .uni-popup__wrapper {
|
||||
height:640rpx;
|
||||
background-color: white !important;
|
||||
border-radius: 24rpx 24rpx 0 0;
|
||||
padding: 29rpx 38rpx 150rpx;
|
||||
}
|
||||
.header {
|
||||
display: flex;
|
||||
height: 56rpx;
|
||||
// padding: 30rpx 38rpx;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
border-radius: 24rpx 24rpx 0 0;
|
||||
|
||||
.header-title {
|
||||
font-size: 32rpx;
|
||||
color: #151515;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.close {
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url('../../static/guanbi.png');
|
||||
}
|
||||
}
|
||||
.title{
|
||||
margin-top: 40rpx;
|
||||
font-size: 26rpx;
|
||||
color: #151515;
|
||||
display: flex;
|
||||
.title-red{
|
||||
color: #E84134;
|
||||
}
|
||||
}
|
||||
.base-price{
|
||||
margin-top: 36rpx;
|
||||
font-size: 26rpx;
|
||||
color: #151515;
|
||||
font-weight: bold;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
}
|
||||
.price-rule{
|
||||
font-size: 24rpx;
|
||||
color: #888888;
|
||||
margin-top: 20rpx;
|
||||
}
|
||||
.billing-weight{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
height: 80rpx;
|
||||
background-color:#F5F5F5 ;
|
||||
border-radius: 4rpx;
|
||||
padding: 10rpx 27rpx;
|
||||
margin-top: 22rpx;
|
||||
.left{
|
||||
font-size: 24prx;
|
||||
color: #888888;
|
||||
}
|
||||
.right{
|
||||
font-size: 26rpx;
|
||||
}
|
||||
}
|
||||
.line{
|
||||
margin-top: 20rpx;
|
||||
width: 100%;
|
||||
height: 2rpx;
|
||||
background-color: #F4F4F4;
|
||||
}
|
||||
.add-service,.setTimeout-send{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
}
|
||||
.add-service{
|
||||
font-size: 26rpx;
|
||||
font-weight: bold;
|
||||
margin-top: 22rpx;
|
||||
margin-bottom: 22rpx;
|
||||
}
|
||||
.setTimeout-send{
|
||||
font-size: 26rpx;
|
||||
.left{
|
||||
color: #888888;
|
||||
}
|
||||
.right{
|
||||
font-weight: bold;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,578 @@
|
||||
<template>
|
||||
<!-- 头部导航栏 -->
|
||||
<nav-bar title='寄快递' :handleToLink="handleToLink"></nav-bar>
|
||||
<view class="express-delivery" v-if="netStatus">
|
||||
<!-- 寄件人,收件人信息填写(地址簿) -->
|
||||
<view class="address-box">
|
||||
<view class="left">
|
||||
<view class="send">寄</view>
|
||||
<view class="line" :class="{active:twoLine>=2}"></view>
|
||||
<view class="get">收</view>
|
||||
</view>
|
||||
<view class="right">
|
||||
<view class="send-people">
|
||||
<view class="send-title" @click="toSendAddressInfo">
|
||||
<view class="send-sub-title">
|
||||
{{senderInfo.name?senderInfo.name:'寄件人信息'}}
|
||||
<text
|
||||
v-if="senderInfo.phone">{{senderInfo.phone.replace(/(\d{3})\d{4}(\d{4})/, "$1****$2")}}</text>
|
||||
</view>
|
||||
<view class="send-desc">
|
||||
{{senderInfo.address?senderInfo.areaLabel+senderInfo.address:'去填写'}}
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<view class="address-enter" @click="handleToAddress('send')">地址簿</view>
|
||||
</view>
|
||||
<view class="line"></view>
|
||||
<view class="get-people">
|
||||
<view class="send-title" @click="toGetAddressInfo">
|
||||
<view class="send-sub-title">
|
||||
{{geterInfo.name?geterInfo.name:'收件人信息'}}
|
||||
<text
|
||||
v-if="geterInfo.phone">{{geterInfo.phone.replace(/(\d{3})\d{4}(\d{4})/, "$1****$2")}}</text>
|
||||
</view>
|
||||
<view class="send-desc">
|
||||
{{geterInfo.address?geterInfo.areaLabel+geterInfo.address:'去填写'}}
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<view class="address-enter" @click="handleToAddress('get')">地址簿</view>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<!-- 寄件其他信息填写 -->
|
||||
<view class="send-form">
|
||||
<view class="tab-box" :class="activeIndex===1?'active':''">
|
||||
<view class="tab-item" :class="activeIndex===index?'active':'isNotActive'"
|
||||
v-for='(item,index) in tabList' @click="changeTab(index)" :key='index'>
|
||||
{{item}}
|
||||
</view>
|
||||
</view>
|
||||
<view class="form-box">
|
||||
<view class="toDoor" v-if='activeIndex === 0'>
|
||||
<!-- 期望上门时间 -->
|
||||
<view class="form-getTime" @click='handleGetTime'>
|
||||
<view class="left">
|
||||
<view class="label">期望上门时间</view>
|
||||
<view class="required"></view>
|
||||
</view>
|
||||
<view class="right">
|
||||
<view class="value" :class="toDoorTimeLabel?'active':''">
|
||||
{{toDoorTimeLabel?toDoorTimeLabel:'请选择'}}
|
||||
</view>
|
||||
<icon class="nextIcon"></icon>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 物品信息 -->
|
||||
<view class="goods-info" @click='handleToGoodsInfo'>
|
||||
<view class="left">
|
||||
<view class="label">物品信息</view>
|
||||
<view class="required"></view>
|
||||
</view>
|
||||
<view class="right">
|
||||
<view class="value" :class="goods.goodsName?'active':''">
|
||||
{{goods.goodsName?(goods.goodsName + ', '+ (goods.weight+'公斤') + (goods.volume?', '+(dealWithVolume(goods.volume/1000000)+'m³'):'')):'请选择'}}
|
||||
</view>
|
||||
<icon class="nextIcon"></icon>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 付款方式 -->
|
||||
<view class="pay-type" @click='handleToPayType'>
|
||||
<view class="left">
|
||||
<view class="label">付款方式</view>
|
||||
</view>
|
||||
<view class="right">
|
||||
<view class="value" :class="payMethod?'active':''">
|
||||
{{payMethod ===1?'寄付':payMethod ===2?'到付':'请选择'}}
|
||||
</view>
|
||||
<icon class="nextIcon"></icon>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 网点自寄 -->
|
||||
<view class="branch-send" v-else>
|
||||
网点自寄
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 期待上门时间弹窗 -->
|
||||
<GetTimePicker ref='timePicker' @getTime="getTime"></GetTimePicker>
|
||||
|
||||
<!-- 付款方式弹窗 -->
|
||||
<PayTypeDialog ref='payType' @getPayType="getPayType"></PayTypeDialog>
|
||||
|
||||
<!-- 底部 -->
|
||||
<view class="footer" :class="{active:isSeachPriceDetail}">
|
||||
<view class="left">
|
||||
<view class="left-top" @click="searchPriceDetail">
|
||||
<view class="all-account">
|
||||
预估总价
|
||||
<view class="price">¥{{feightPrice.expense}}</view>
|
||||
</view>
|
||||
<view class="price-detail">
|
||||
明细
|
||||
<view class="arrow"></view>
|
||||
</view>
|
||||
</view>
|
||||
<view class="left-bottom">
|
||||
<view class="active"></view>
|
||||
<view class="content">阅读并同意《电子运单契约条款》</view>
|
||||
</view>
|
||||
</view>
|
||||
<view class="right">
|
||||
<view class="btn" :class="isCanSubmit?'active':''" @click="order">下单</view>
|
||||
|
||||
</view>
|
||||
</view>
|
||||
<!-- 运费明细弹窗 -->
|
||||
<uni-popup ref="priceDetail" type="bottom" class="priceDetail">
|
||||
<view class="header">
|
||||
<view class="header-title">运费预估</view>
|
||||
<view class="close" @click="handleCancel"></view>
|
||||
</view>
|
||||
<view class="title">
|
||||
实际费用请以快递员揽收为准,
|
||||
<view class="title-red" @click="openAccountRulesDialog">了解计费规则</view>
|
||||
</view>
|
||||
<view class="base-price">
|
||||
<view class="left">基础运费</view>
|
||||
<view class="right">¥{{feightPrice.expense}}</view>
|
||||
</view>
|
||||
<view class="price-rule">首重(1.0kg){{feightPrice.firstWeight}}元,续重{{feightPrice.continuousWeight}}元/kg</view>
|
||||
<view class="billing-weight">
|
||||
<view class="left">计费重量</view>
|
||||
<view class="right">(含包装){{feightPrice.weight}}kg</view>
|
||||
</view>
|
||||
<view class="line"></view>
|
||||
<view class="add-service">
|
||||
<view class="left">增值服务</view>
|
||||
<view class="right">¥0</view>
|
||||
</view>
|
||||
<view class="setTimeout-send">
|
||||
<view class="left">礼物定时派送</view>
|
||||
<view class="right">¥0</view>
|
||||
</view>
|
||||
</uni-popup>
|
||||
</view>
|
||||
<!-- 断网显示的页面 -->
|
||||
<net-fail v-else :handleToRefresh="handleToRefresh"></net-fail>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
reactive,
|
||||
onMounted,
|
||||
computed,
|
||||
nextTick
|
||||
} from 'vue';
|
||||
import {
|
||||
onLoad,
|
||||
} from '@dcloudio/uni-app';
|
||||
import {
|
||||
getAddressInfoDetail,
|
||||
defaultAddress
|
||||
} from '@/pages/api/address.js'
|
||||
import {
|
||||
doOrder,
|
||||
getEstimatePrice
|
||||
} from '@/pages/api/order.js'
|
||||
import GetTimePicker from './components/getTimePicker.vue';
|
||||
import PayTypeDialog from './components/payType.vue';
|
||||
import {
|
||||
handleSecondQi,
|
||||
} from '@/utils/index.js'
|
||||
import {
|
||||
useStore
|
||||
} from 'vuex';
|
||||
const twoLine = ref()
|
||||
const store = useStore(); //vuex获取、储存数据
|
||||
const users = store.state.user
|
||||
const isCanSubmit = computed(() => {
|
||||
|
||||
return (geterInfo.name && senderInfo.name && goods.goodsName && payMethod.value && toDoorTimeLabel.value)
|
||||
})
|
||||
|
||||
const timePicker = ref() //期望上门时间组件的引用
|
||||
const payType = ref() //付款方式弹窗组件的引用
|
||||
const priceDetail = ref()//运费弹窗明细组件的引用
|
||||
const isSeachPriceDetail = ref(false)
|
||||
const toDoorTime = ref('') //预计上门时间
|
||||
const toDoorTimeLabel = ref('今天 一小时内')
|
||||
const payMethod = ref(1) //付款方式
|
||||
const type = ref('') //判断是从寄件地址返回的还是收件地址
|
||||
const feightPrice = reactive({
|
||||
expense: '', //运费
|
||||
weight: '', //计算重量
|
||||
firstWeight: '', //首重价格
|
||||
continuousWeight: '' //续重价格
|
||||
})
|
||||
const isFromGoods = ref(false)
|
||||
const stopClick = ref(false)
|
||||
const goods = reactive({
|
||||
weight: 1,
|
||||
volume: 1,
|
||||
goodsName: '',
|
||||
goodsType: ''
|
||||
})
|
||||
const sendAddress = ref('')
|
||||
const senderInfo = reactive({
|
||||
name: '',
|
||||
phone: '',
|
||||
address: '',
|
||||
areaLabel: '',
|
||||
}) //寄件人信息
|
||||
const geterInfo = reactive({
|
||||
name: '',
|
||||
phone: '',
|
||||
address: '',
|
||||
areaLabel: '',
|
||||
}) //收件人信息
|
||||
let netStatus = ref(true)
|
||||
//tab常量
|
||||
const tabList = reactive(['上门取件',
|
||||
'网点自寄'
|
||||
])
|
||||
//当前
|
||||
let activeIndex = ref(0)
|
||||
//打开计费规则弹窗
|
||||
const openAccountRulesDialog = () => {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/account-rules/index'
|
||||
});
|
||||
}
|
||||
onMounted(() => {
|
||||
feightPrice.expense = users.expense
|
||||
feightPrice.weight = users.computeWeight
|
||||
feightPrice.firstWeight = users.firstWeight
|
||||
feightPrice.continuousWeight = users.continuousWeight
|
||||
})
|
||||
onLoad((options) => {
|
||||
console.log(options,'options')
|
||||
if (options.type === 'send') {
|
||||
sendAddress.value = options.id
|
||||
uni.setStorageSync('sendId', options.id)
|
||||
} else if (options.type === 'get') {
|
||||
uni.setStorageSync('getId', options.id)
|
||||
}
|
||||
//用于解决点击地址薄选择了非默认地址回来之后被默认地址覆盖的问题
|
||||
if(!options.id){
|
||||
getDefaultAddress()
|
||||
}
|
||||
|
||||
//用于解决页面跳转回来数据缺失问题
|
||||
if (uni.getStorageSync('sendId')) {
|
||||
searchAddressDetail(uni.getStorageSync('sendId'), 'send')
|
||||
}
|
||||
if (uni.getStorageSync('getId')) {
|
||||
searchAddressDetail(uni.getStorageSync('getId'), 'get')
|
||||
}
|
||||
type.value = options.type
|
||||
goods.volume = users.volume
|
||||
goods.weight = users.weight
|
||||
goods.goodsName = users.goodsInfo.name
|
||||
goods.goodsType = users.goodsInfo.goodsType ? users.goodsInfo.goodsType.id : ''
|
||||
toDoorTimeLabel.value = users.toDoorTimeLabel || '今天 一小时内'
|
||||
toDoorTime.value = users.toDoorTime
|
||||
payMethod.value = users.payMethod
|
||||
isFromGoods.value = options.isFromGoods
|
||||
//必须寄件收件人地址和物品信息都选择了才能计算运费
|
||||
//这里不能将逻辑提前,应该置于赋值后面
|
||||
if (uni.getStorageSync('getId') && (uni.getStorageSync('sendId') || sendAddress.value) && users.goodsInfo
|
||||
.name) {
|
||||
console.log(666)
|
||||
calcFreight()
|
||||
}
|
||||
})
|
||||
const handleToRefresh = ()=>{
|
||||
uni.redirectTo({
|
||||
url:'/pages/express-delivery/index'
|
||||
})
|
||||
}
|
||||
//处理体积如果是超过小数点后四位则保留四位小数,其他情况直接展示
|
||||
const dealWithVolume = (value) => {
|
||||
let dot = String(value).indexOf(".");
|
||||
if (dot != -1) {
|
||||
let dotCnt = String(value).substring(dot + 1, value.length);
|
||||
if (dotCnt.length > 4) {
|
||||
value = value.toFixed(4);
|
||||
return value
|
||||
} else {
|
||||
return value
|
||||
}
|
||||
} else {
|
||||
return value
|
||||
}
|
||||
}
|
||||
const getDefaultAddress = () => {
|
||||
if (!type.value && !isFromGoods.value) {
|
||||
defaultAddress().then((res) => {
|
||||
if (res.data) {
|
||||
sendAddress.value = res.data.id
|
||||
senderInfo.name = res.data.name
|
||||
senderInfo.phone = res.data.phoneNumber
|
||||
senderInfo.address = res.data.address
|
||||
senderInfo.areaLabel = res.data.province.name + ' ' + res.data.city.name + ' ' + res.data
|
||||
.county.name
|
||||
uni.setStorageSync('sendId', res.data.id)
|
||||
netStatus.value = true
|
||||
nextTick(()=>{
|
||||
//用来获取详细地址有几行,从而适配寄和收之间竖线的高度
|
||||
uni.createSelectorQuery().select('.send-desc').boundingClientRect(res => {
|
||||
let height = res.height;
|
||||
let line = height / 15
|
||||
twoLine.value = line
|
||||
}).exec();
|
||||
})
|
||||
}
|
||||
}).catch((err) => {
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
});
|
||||
netStatus.value = false
|
||||
})
|
||||
}
|
||||
|
||||
}
|
||||
//查询地址簿详情,并赋值
|
||||
const searchAddressDetail = (id, type) => {
|
||||
getAddressInfoDetail(id).then((res) => {
|
||||
const {
|
||||
name,
|
||||
phoneNumber,
|
||||
address,
|
||||
city,
|
||||
county,
|
||||
province,
|
||||
} = res.data
|
||||
if (type === 'send') {
|
||||
senderInfo.name = name
|
||||
senderInfo.phone = phoneNumber
|
||||
senderInfo.address = address
|
||||
senderInfo.areaLabel = province.name + ' ' + city.name + ' ' + county.name
|
||||
} else {
|
||||
geterInfo.name = name
|
||||
geterInfo.phone = phoneNumber
|
||||
geterInfo.address = address
|
||||
geterInfo.areaLabel = province.name + ' ' + city.name + ' ' + county.name
|
||||
}
|
||||
nextTick(()=>{
|
||||
//用来获取详细地址有几行,从而适配寄和收之间竖线的高度
|
||||
uni.createSelectorQuery().select('.send-desc').boundingClientRect(res => {
|
||||
let height = res.height;
|
||||
let line = height / 15
|
||||
twoLine.value = line
|
||||
}).exec();
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
// 切换tab
|
||||
const changeTab = (index) => {
|
||||
if (index === 1) {
|
||||
return handleSecondQi()
|
||||
}
|
||||
activeIndex.value = index
|
||||
}
|
||||
|
||||
//打开选择期望上门时间弹窗
|
||||
const handleGetTime = () => {
|
||||
timePicker.value.handleOpen()
|
||||
}
|
||||
//打开付款方式弹窗
|
||||
const handleToPayType = () => {
|
||||
payType.value.handleOpen()
|
||||
}
|
||||
const clearCurrentPageData = () => {
|
||||
store.commit('user/setGoodsInfo', {})
|
||||
store.commit('user/setVolume', '')
|
||||
store.commit('user/setExpense', 0)
|
||||
store.commit('user/setComputeWeight', 0)
|
||||
store.commit('user/setFirstWeight', 0)
|
||||
store.commit('user/setContinuousWeight', 0)
|
||||
store.commit('user/setToDoorTimeLabel', '')
|
||||
store.commit('user/setToDoorTime', String(new Date().getFullYear()) + '-' + String(new Date().getMonth() + 1) +
|
||||
'-' +
|
||||
new Date().getDate() + ' ' + String(new Date().getHours() + 1) + ':' + String(Number(new Date()
|
||||
.getMinutes()) < 10 ? '0' + Number(new Date().getMinutes()) : Number(new Date()
|
||||
.getMinutes())) + ':00')
|
||||
store.commit('user/setWeight', 1)
|
||||
store.commit('user/setPayMethod', 1)
|
||||
}
|
||||
|
||||
const handleToLink = () => {
|
||||
clearCurrentPageData()
|
||||
uni.removeStorageSync('getId');
|
||||
uni.removeStorageSync('sendId');
|
||||
uni.switchTab({
|
||||
url: '/pages/index/index'
|
||||
});
|
||||
}
|
||||
|
||||
//跳转到物品信息页面
|
||||
const handleToGoodsInfo = () => {
|
||||
uni.navigateTo({
|
||||
url: '/pages/goodsInfo/index'
|
||||
});
|
||||
}
|
||||
|
||||
//跳转地址簿
|
||||
const handleToAddress = (type) => {
|
||||
uni.navigateTo({
|
||||
url: '/pages/address/index?type=' + type + '&isFromAddress=false'
|
||||
});
|
||||
}
|
||||
|
||||
//查看运费明细
|
||||
const searchPriceDetail = () => {
|
||||
priceDetail.value.open('bottom')
|
||||
isSeachPriceDetail.value = true
|
||||
}
|
||||
|
||||
//关闭运费明细弹窗
|
||||
const handleCancel = () => {
|
||||
priceDetail.value.close()
|
||||
isSeachPriceDetail.value = false
|
||||
}
|
||||
|
||||
//重置期望上门时间为最新时间
|
||||
const resetPickUpTime = ()=>{
|
||||
if(toDoorTimeLabel.value === '今天 一小时内'){
|
||||
store.commit('user/setToDoorTime', toDoorTime.value)
|
||||
}
|
||||
}
|
||||
|
||||
//下单
|
||||
const order = () => {
|
||||
if (!isCanSubmit.value) {
|
||||
return
|
||||
}
|
||||
if (stopClick.value) {
|
||||
return
|
||||
}
|
||||
stopClick.value = true
|
||||
console.log(goods,'下单')
|
||||
doOrder({
|
||||
goodNum: 1,
|
||||
goodsName: goods.goodsName,
|
||||
goodsType: goods.goodsType,
|
||||
payMethod: payMethod.value,
|
||||
pickUpTime:toDoorTimeLabel.value==='今天 一小时内'?String(new Date().getFullYear()) + '-' + String(new Date().getMonth() + 1) + '-' +
|
||||
new Date().getDate() + ' ' + String(new Date().getHours() + 1) + ':' + String(Number(new Date()
|
||||
.getMinutes()) < 10 ? '0' + Number(new Date().getMinutes()) : Number(new Date()
|
||||
.getMinutes())) + ':00': users.toDoorTime,
|
||||
totalVolume: goods.volume ,
|
||||
totalWeight: goods.weight,
|
||||
receiptAddress: uni.getStorageSync('getId'),
|
||||
sendAddress: sendAddress.value || uni.getStorageSync('sendId'),
|
||||
pickupType: 0 //0上门取件,1网点自寄
|
||||
}).then((res) => {
|
||||
console.log(res,'resss')
|
||||
if (res.code !== 200) {
|
||||
uni.showToast({
|
||||
title: res.data.msg,
|
||||
icon: 'none',
|
||||
duration: 1000
|
||||
})
|
||||
} else {
|
||||
|
||||
uni.redirectTo({
|
||||
url: '/subPages/order-success/index?orderId=' + res.data.id
|
||||
});
|
||||
uni.removeStorageSync('getId');
|
||||
uni.removeStorageSync('sendId');
|
||||
clearCurrentPageData()
|
||||
}
|
||||
stopClick.value = false
|
||||
}).catch((err)=>{
|
||||
stopClick.value = false
|
||||
console.log(err,'err')
|
||||
})
|
||||
|
||||
}
|
||||
|
||||
//跳转到寄件人地址信息填写页面
|
||||
const toSendAddressInfo = () => {
|
||||
if (sendAddress.value || uni.getStorageSync('sendId')) {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/address-info/index?type=send&editOrAdd=edit' + '&id=' +
|
||||
uni.getStorageSync(
|
||||
'sendId')
|
||||
});
|
||||
} else {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/address-info/index?type=send&editOrAdd=add'
|
||||
});
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
//跳转到收件人地址信息填写页面
|
||||
const toGetAddressInfo = () => {
|
||||
if (uni.getStorageSync('getId')) {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/address-info/index?type=get&editOrAdd=edit' + '&id=' + uni.getStorageSync(
|
||||
'getId')
|
||||
});
|
||||
} else {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/address-info/index?type=get&editOrAdd=add'
|
||||
});
|
||||
}
|
||||
}
|
||||
//获取期望上门时间数据
|
||||
const getTime = (value) => {
|
||||
|
||||
toDoorTimeLabel.value = value.selectedDayLabel + ' ' + value.selectedTimeLabel
|
||||
toDoorTime.value = String(new Date().getFullYear()) + '-' + String(new Date().getMonth() + 1) + '-' + (
|
||||
new Date().getDate() + value.selectedDay + ' ' + String((value.selectedTime === 1 ? new Date()
|
||||
.getHours() + 1 : value.selectedTime) + ':00:00'))
|
||||
store.commit('user/setToDoorTimeLabel', toDoorTimeLabel.value)
|
||||
store.commit('user/setToDoorTime', toDoorTime.value)
|
||||
}
|
||||
//获取付款方式数据
|
||||
const getPayType = (value) => {
|
||||
payMethod.value = value
|
||||
store.commit('user/setPayMethod', value)
|
||||
}
|
||||
//计算运费
|
||||
const calcFreight = () => {
|
||||
console.log(goods,'计算运费')
|
||||
getEstimatePrice({
|
||||
goodNum: 1,
|
||||
goodsName: goods.goodsName,
|
||||
goodsType: goods.goodsType,
|
||||
payMethod: payMethod.value,
|
||||
pickUpTime: toDoorTime.value,
|
||||
totalVolume: goods.volume ,
|
||||
totalWeight: goods.weight,
|
||||
receiptAddress: uni.getStorageSync('getId'),
|
||||
sendAddress: uni.getStorageSync('sendId'),
|
||||
pickupType: 0 //0上门取件,1网点自寄
|
||||
}).then((res) => {
|
||||
if (res.code == 200) {
|
||||
store.commit('user/setExpense', res.data.expense)
|
||||
store.commit('user/setComputeWeight', res.data.computeWeight)
|
||||
store.commit('user/setFirstWeight', res.data.firstWeight)
|
||||
store.commit('user/setContinuousWeight', res.data.continuousWeight)
|
||||
feightPrice.expense = res.data.expense
|
||||
feightPrice.weight = res.data.computeWeight
|
||||
feightPrice.firstWeight = res.data.firstWeight
|
||||
feightPrice.continuousWeight = res.data.continuousWeight
|
||||
} else {
|
||||
uni.showToast({
|
||||
title: res.msg,
|
||||
duration: 1000,
|
||||
icon: 'none'
|
||||
});
|
||||
}
|
||||
})
|
||||
}
|
||||
</script>
|
||||
<style src='./components/getTimePicker.scss' lang="scss"></style>
|
||||
<style src='./components/payType.scss' lang="scss"></style>
|
||||
<style src="./index.scss" lang="scss"></style>
|
@@ -0,0 +1,179 @@
|
||||
.goodsSearch{
|
||||
background-color: white;
|
||||
padding: 38rpx 40rpx 0rpx 40rpx;
|
||||
.searchBox-title{
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
.title{
|
||||
font-size: 32rpx;
|
||||
color: #0F0F0F;
|
||||
font-weight: bold;
|
||||
.label{
|
||||
display: inline-block;
|
||||
width: 62rpx;
|
||||
height: 30rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url('../../../static/required.png');
|
||||
margin-left: 10rpx;
|
||||
}
|
||||
}
|
||||
.stopSend{
|
||||
font-size: 24rpx;
|
||||
color: #888888;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
.gantanhao{
|
||||
display: inline-block;
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url('../../../static/gantanhao.png');
|
||||
}
|
||||
}
|
||||
}
|
||||
.selected-goods{
|
||||
padding-top: 32rpx;
|
||||
overflow: hidden;
|
||||
.goods{
|
||||
position: relative;
|
||||
margin-bottom: 46rpx;
|
||||
border: 2rpx solid #E63E32;
|
||||
border-radius: 32rpx;
|
||||
font-size: 24rpx;
|
||||
color:#E63E32 ;
|
||||
text-align: center;
|
||||
line-height: 34rpx;
|
||||
padding:15rpx 22rpx 15rpx 18rpx ;
|
||||
height: 34rpx;
|
||||
display: inline-block;
|
||||
image{
|
||||
position: absolute;
|
||||
right: 0rpx;
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
bottom: 10rpx;
|
||||
top: -20rpx;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
.seachBox{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
::v-deep .uni-searchbar{
|
||||
padding-left: 0rpx!important;
|
||||
.uni-searchbar__box{
|
||||
border-radius: 34rpx!important;
|
||||
justify-content: flex-start;
|
||||
}
|
||||
|
||||
}
|
||||
uni-search-bar{
|
||||
flex: auto;
|
||||
}
|
||||
.search-icon{
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url('../../../static/search.png');
|
||||
}
|
||||
.search-btn{
|
||||
font-weight: bold;
|
||||
font-size: 28rpx;
|
||||
}
|
||||
}
|
||||
.recent-send,.recent-send-goods,.hot-send-goods{
|
||||
padding: 20rpx 40rpx 0rpx 0rpx;
|
||||
.recent-send-title{
|
||||
font-size: 26rpx;
|
||||
color: #0F0F0F;
|
||||
}
|
||||
.recent-send-list{
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
padding-bottom: 44rpx;
|
||||
.recent-send-item{
|
||||
margin-right: 20rpx;
|
||||
margin-top: 35rpx;
|
||||
font-size: 24rpx;
|
||||
color: #0F0F0F;
|
||||
padding:15rpx 22rpx 15rpx 18rpx ;
|
||||
height: 34rpx;
|
||||
background: #F4F4F4;
|
||||
border-radius: 32rpx;
|
||||
text-align: center;
|
||||
line-height: 34rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.recent-send-goods{
|
||||
.title{
|
||||
font-size: 32rpx;
|
||||
color: #0F0F0F;
|
||||
font-weight: bold;
|
||||
}
|
||||
}
|
||||
.hot-send-goods{
|
||||
font-size: 28rpx;
|
||||
padding-top: 0rpx;
|
||||
.title{
|
||||
font-size: 28rpx;
|
||||
color: #0F0F0F;
|
||||
font-weight: bold;
|
||||
}
|
||||
}
|
||||
.search-list{
|
||||
.seatch-list-item{
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
height: 102rpx;
|
||||
border-bottom: 2rpx solid #F4F4F4;
|
||||
.label{
|
||||
font-size: 28rpx;
|
||||
color: #151515;
|
||||
}
|
||||
.classify{
|
||||
font-size: 28rpx;
|
||||
color: #888888;
|
||||
}
|
||||
}
|
||||
}
|
||||
::v-deep .uni-popup{
|
||||
.uni-popup__wrapper{
|
||||
background-color:white!important;
|
||||
padding:36rpx 33rpx 43rpx 33rpx ;
|
||||
border-radius: 24rpx 24rpx 0 0;
|
||||
}
|
||||
.title{
|
||||
font-size: 30rpx;
|
||||
color: #151515;
|
||||
text-align: center;
|
||||
margin-bottom: 34rpx;
|
||||
font-weight: bold;
|
||||
}
|
||||
.content{
|
||||
font-size: 28rpx;
|
||||
color: #151515;
|
||||
line-height: 40prx;
|
||||
margin-top: 28rpx;
|
||||
}
|
||||
.btn{
|
||||
width: 404rpx;
|
||||
height: 88rpx;
|
||||
background-color: #E84134;
|
||||
border-radius: 44rpx;
|
||||
text-align: center;
|
||||
line-height: 88rpx;
|
||||
color: white;
|
||||
margin: 0 auto;
|
||||
margin-top: 42rpx;
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,270 @@
|
||||
<!-- 物品类型弹窗页面 -->
|
||||
<template>
|
||||
<view class="goodsSearch">
|
||||
<!-- 头部 -->
|
||||
<view class="searchBox-title">
|
||||
<view class="title">
|
||||
物品名称
|
||||
<view class="label"></view>
|
||||
</view>
|
||||
<view class="stopSend" @click="openForbidGoodsDialog">
|
||||
禁寄物品
|
||||
<view class="gantanhao"></view>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 搜索框 -->
|
||||
<view class="seachBox" v-if="!isSelectedGoods">
|
||||
<uni-search-bar :cancelButton='isFocus && searchValue?"none":"auto"' :focus="isFocus" placeholder="请输入搜索内容"
|
||||
v-model="searchValue" @blur="blur" @focus="focus" @input="input" @cancel="cancel" @clear="clear">
|
||||
<template v-slot:searchIcon>
|
||||
<view class="search-icon"></view>
|
||||
</template>
|
||||
</uni-search-bar>
|
||||
<view class="search-btn" @click="userDefined" v-if="searchValue && isFocus">确定</view>
|
||||
</view>
|
||||
|
||||
<!-- 最近寄件 -->
|
||||
<view class="recent-send-goods" v-if='!isSelectedGoods && recentSendList.data.length && !isInput'>
|
||||
<view class="title">最近寄件</view>
|
||||
<view class="recent-send-list">
|
||||
<view class="recent-send-item" v-for="(item,index) in recentSendList.data" :key='index'
|
||||
@click="handleClick(item)">
|
||||
{{item.name}}{{item.goodsType?'('+item.goodsType.name+')':''}}
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 选择的物品 -->
|
||||
<view class="selected-goods" v-if="isSelectedGoods">
|
||||
<view class="goods" @click.prevent="handleSelectedGoods">
|
||||
<image src='../../../static/shanchu-after.png' @click.stop="handleCancelGood"></image>
|
||||
{{goods.info.name}}{{goods.info.goodsType?'('+goods.info.goodsType.name+')':""}}
|
||||
</view>
|
||||
|
||||
|
||||
</view>
|
||||
|
||||
<!-- 热门寄件 -->
|
||||
<view class="hot-send-goods" v-if='!isSelectedGoods && !isInput && isFocus'>
|
||||
<view class="title">热门寄件</view>
|
||||
<view class="recent-send-list">
|
||||
<view class="recent-send-item" v-for="(item,index) in hotSendList.data" :key='index'
|
||||
@click="handleClick(item)">
|
||||
{{item.name}}{{item.goodsType?'('+item.goodsType.name+')':''}}
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<!-- 模糊查询出的列表 -->
|
||||
<view class="search-list" v-if="isInput">
|
||||
<view class="seatch-list-item" @click="handleClick(item)" v-for="(item,index) in dimSearchList.data"
|
||||
:key="index">
|
||||
<view class="label">{{item.name}}</view>
|
||||
<view class="classify">{{item.name}}{{item.goodsType?'('+item.goodsType.name+')':''}}</view>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<!-- 禁寄物品弹窗 -->
|
||||
<uni-popup ref="popup" type="bottom" :safe-area="false">
|
||||
<view class="title">禁止寄递物品目录</view>
|
||||
<view class="content">
|
||||
1.枪支(含仿制品、主要零部件)弹药。
|
||||
</view>
|
||||
<view class="content">
|
||||
2.管制器具。如匕首、三棱刮刀、带有自锁装置的弹簧刀(跳刀)、弩、催泪器等。
|
||||
</view>
|
||||
<view class="content">
|
||||
3.爆炸物品。如炸药、雷管、导火索、烟花、鞭炮、摔炮、拉炮、砸炮等。
|
||||
</view>
|
||||
<view class="content">
|
||||
4.压缩和液化气体及其容器。如氢气、甲烷、乙烷、乙炔、打火机、氯气、压缩氧气、氮气等。
|
||||
</view>
|
||||
<view class="content">
|
||||
5.易燃液体。如汽油、柴油、煤油、桐油、丙酮、乙醚、油漆、生漆、苯、酒精、松香油等。
|
||||
</view>
|
||||
<view class="content">
|
||||
6.易燃固体、自燃物质、遇水易燃物质。如红磷、硫磺、固体酒精、火柴、活性炭等。
|
||||
</view>
|
||||
<view class="content">
|
||||
7.氧化剂和过氧化物。如高锰酸盐、高氯酸盐、氧化氢、双氧水等。
|
||||
</view>
|
||||
<view class="content">
|
||||
8.毒性物质。如砷、砒霜、汞化物、铊化物、氰化物、硒粉、苯酚、汞、剧毒农药等。
|
||||
</view>
|
||||
<view class="content">
|
||||
9.生化制品、传染性、感染性物质。如病菌、炭疽、寄生虫、排泄物、医疗废弃物、尸骨、动物器官等。
|
||||
</view>
|
||||
<view class="content">
|
||||
10.放射性物质。如铀、钴、镭、钚等。
|
||||
</view>
|
||||
|
||||
<view class="btn" @click="closeForbidGoodsDialog">知道了</view>
|
||||
</uni-popup>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
reactive,
|
||||
onMounted,
|
||||
} from 'vue';
|
||||
import {
|
||||
usualGoodsList,
|
||||
goodsList
|
||||
} from '@/pages/api/order.js'
|
||||
import {
|
||||
useStore
|
||||
} from 'vuex';
|
||||
const store = useStore(); //vuex获取、储存数据
|
||||
const users = store.state.user
|
||||
const emits = defineEmits(["@getGoodsInfo"]);
|
||||
const props = defineProps({
|
||||
isShowOther: {
|
||||
type: Function,
|
||||
required: true
|
||||
},
|
||||
})
|
||||
const goods = reactive({
|
||||
info: {}
|
||||
})
|
||||
const isSelectedGoods = ref(false) //是否选择了物品
|
||||
const popup = ref()
|
||||
const searchValue = ref('')
|
||||
const isFocus = ref(false) //是否搜索框为聚焦状态
|
||||
const isInput = ref(false) //是否搜索框为输入状态
|
||||
const isUpload = ref(true) //是否输入框加载完毕
|
||||
//模糊查询出的列表数据
|
||||
const dimSearchList = reactive({
|
||||
data: []
|
||||
})
|
||||
//热门寄件物品
|
||||
const hotSendList = reactive({
|
||||
data: []
|
||||
})
|
||||
|
||||
const recentSendList = reactive({
|
||||
data: [],
|
||||
})
|
||||
|
||||
onMounted(() => {
|
||||
getData()
|
||||
})
|
||||
|
||||
const getData = () => {
|
||||
usualGoodsList({
|
||||
name: ''
|
||||
}).then((res) => {
|
||||
if (res.data) {
|
||||
hotSendList.data = res.data.slice(0, 6)
|
||||
}
|
||||
}).catch((err) => {
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
});
|
||||
})
|
||||
isSelectedGoods.value = users.goodsInfo.name
|
||||
if (isSelectedGoods.value) goods.info = users.goodsInfo
|
||||
|
||||
goodsList().then((res) => {
|
||||
if (res.data) {
|
||||
recentSendList.data = res.data.slice(0, 5)
|
||||
}
|
||||
}).catch((err) => {
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
});
|
||||
})
|
||||
}
|
||||
const handleSelectedGoods = () => {
|
||||
searchValue.value = goods.info.name
|
||||
isFocus.value = true
|
||||
isInput.value = true
|
||||
isSelectedGoods.value = false
|
||||
dimSearch(searchValue.value)
|
||||
props.isShowOther(true, 'always')
|
||||
}
|
||||
//模糊查询列表数据
|
||||
const dimSearch = (key) => {
|
||||
if (!isUpload.value) return
|
||||
isUpload.value = false
|
||||
usualGoodsList({
|
||||
name: key
|
||||
}).then((res) => {
|
||||
dimSearchList.data = res.data
|
||||
isUpload.value = true
|
||||
})
|
||||
}
|
||||
|
||||
const input = (e) => {
|
||||
isFocus.value = e ? true : false
|
||||
isInput.value = Boolean(e)
|
||||
if (!isUpload.value) return
|
||||
searchValue.value = e
|
||||
dimSearch(e)
|
||||
props.isShowOther(e ? true : false)
|
||||
}
|
||||
const cancel = () => {
|
||||
isInput.value = false
|
||||
props.isShowOther(false)
|
||||
}
|
||||
const clear = () => {
|
||||
props.isShowOther(true)
|
||||
}
|
||||
const blur = () => {
|
||||
console.log('blur')
|
||||
}
|
||||
const focus = () => {
|
||||
isFocus.value = true
|
||||
props.isShowOther(true)
|
||||
}
|
||||
|
||||
//用户自定义的物品
|
||||
const userDefined = () => {
|
||||
goods.info = {
|
||||
name: searchValue.value
|
||||
}
|
||||
isSelectedGoods.value = Boolean(searchValue.value)
|
||||
props.isShowOther(false)
|
||||
emits('getGoodsInfo', {
|
||||
name: searchValue.value
|
||||
})
|
||||
isFocus.value = false
|
||||
}
|
||||
//取消物品选择
|
||||
const handleCancelGood = () => {
|
||||
goods.info = {}
|
||||
store.commit('user/setGoodsInfo', {})
|
||||
emits('getGoodsInfo', {})
|
||||
isSelectedGoods.value = false
|
||||
searchValue.value = ''
|
||||
isFocus.value = false
|
||||
isInput.value = false
|
||||
}
|
||||
|
||||
//点击标签物品
|
||||
const handleClick = (item) => {
|
||||
searchValue.value = item.value
|
||||
isFocus.value = true
|
||||
goods.info = item
|
||||
isSelectedGoods.value = true
|
||||
isInput.value = false
|
||||
props.isShowOther(false)
|
||||
emits('getGoodsInfo', item)
|
||||
}
|
||||
|
||||
//打开禁寄物品弹窗
|
||||
const openForbidGoodsDialog = () => {
|
||||
popup.value.open('bottom');
|
||||
}
|
||||
|
||||
//关闭禁寄物品弹窗
|
||||
const closeForbidGoodsDialog = () => {
|
||||
popup.value.close();
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped src='./goodsSearch.scss'></style>
|
@@ -0,0 +1,109 @@
|
||||
.weightAndVolume{
|
||||
background-color: white;
|
||||
margin-top: 20rpx;
|
||||
padding:56rpx 40rpx ;
|
||||
position: relative;
|
||||
z-index: 1;
|
||||
margin-bottom: 200rpx;
|
||||
.volume-box{
|
||||
margin-top: 50rpx;
|
||||
.isVolumeInfo{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
.isVolumeInfo-title{
|
||||
font-size: 24rpx;
|
||||
color: #888888;
|
||||
margin-right: 18rpx;
|
||||
}
|
||||
::v-deep .uni-switch-input::before{
|
||||
background-color: #CCCCCC;
|
||||
width: 120rpx;
|
||||
}
|
||||
::v-deep .uni-switch-input.uni-switch-input-checked:after{
|
||||
transform: translateX(60rpx);
|
||||
}
|
||||
::v-deep .uni-switch-input{
|
||||
width: 120rpx;
|
||||
}
|
||||
switch{
|
||||
position: relative;
|
||||
left: 20rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
.weight-box,.volume-box,.isVolumeInfo-title-box{
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
position: relative;
|
||||
.unit{
|
||||
position: absolute;
|
||||
right: 126rpx;
|
||||
top:16rpx ;
|
||||
}
|
||||
.weight-box-title,.volume-box-title{
|
||||
font-weight: bold;
|
||||
font-size: 32rpx;
|
||||
color: #0F0F0F;
|
||||
}
|
||||
.uni-numbox{
|
||||
height: 76rpx;
|
||||
.uni-numbox__value{
|
||||
height: 76rpx!important;
|
||||
margin: 0;
|
||||
background-color: #F9F9F9!important;
|
||||
width: 220rpx;
|
||||
}
|
||||
.uni-numbox__plus,.uni-numbox__minus{
|
||||
width: 76rpx;
|
||||
border-radius: 0rpx 36rpx 36rpx 0;
|
||||
background-color: #ECECEC!important;
|
||||
.uni-numbox--text{
|
||||
span{
|
||||
font-weight: bold;
|
||||
}
|
||||
}
|
||||
}
|
||||
.uni-numbox__minus{
|
||||
border-radius: 36rpx 0rpx 0rpx 36rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
.isVolumeInfo-title-box{
|
||||
justify-content: flex-end;
|
||||
margin-top: 10rpx;
|
||||
.unit{
|
||||
|
||||
}
|
||||
}
|
||||
.long-width-height{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
margin-top: 16rpx;
|
||||
.cheng{
|
||||
margin: 0 12rpx;
|
||||
height: 19rpx;
|
||||
line-height: 30rpx;
|
||||
}
|
||||
.long,.width,.height{
|
||||
::v-deep .uni-input-wrapper,.uni-input{
|
||||
background-color: #F4F4F4;
|
||||
border-radius: 36rpx;
|
||||
width: 198rpx;
|
||||
height: 74rpx;
|
||||
text-align: center;
|
||||
}
|
||||
}
|
||||
}
|
||||
.accounting-rules{
|
||||
font-size: 24rpx;
|
||||
color: #888888;
|
||||
margin-top: 40rpx;
|
||||
.red{
|
||||
color: #E84134;
|
||||
font-size: 24rpx;
|
||||
text-decoration: underline;
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,244 @@
|
||||
<!-- 重量和体积 -->
|
||||
<template>
|
||||
<view class="weightAndVolume">
|
||||
<!-- 重量 -->
|
||||
<view class="weight-box">
|
||||
<view class="weight-box-title">预估重量</view>
|
||||
<!-- <uni-number-box max="9999" min="1" :value="weight" @change="changeWeight" /> -->
|
||||
<view class="number-box">
|
||||
<view class="minus-btn" :class="isLessThan ? 'active' : ''" @click="handleMinus">-</view>
|
||||
<input class="uni-input" type="digit" maxlength="4" v-model="weight" @blur="handleWeigthBlur"/>
|
||||
<view class="add-btn" :class="isExceed ? 'active' : ''" @click="handleAdd">+</view>
|
||||
</view>
|
||||
<view class="unit">kg</view>
|
||||
</view>
|
||||
<!-- 体积 -->
|
||||
<view class="volume-box">
|
||||
<view class="volume-box-title">总体积</view>
|
||||
<view class="isVolumeInfo">
|
||||
<view class="isVolumeInfo-title">补充体积预估费用更准确</view>
|
||||
<switch color="#1DC779" style="transform:scale(0.7)" @change="switchChange" :checked="isVolumeInfo" />
|
||||
</view>
|
||||
</view>
|
||||
<!-- 体积输入框 -->
|
||||
<view class="isVolumeInfo-title-box" v-if='isVolumeInfo'>
|
||||
<uni-number-box />
|
||||
<view class="number-box">
|
||||
<view class="minus-btn" :class="isLessThanVolume ? 'active' : ''" @click="handleVolumeMinus">-</view>
|
||||
<input class="uni-input" type="digit" maxlength="6" v-model="volume" @blur="handleVolume" />
|
||||
<view class="add-btn" :class="isExceedVolume ? 'active' : ''" @click="handleVolumeAdd">+</view>
|
||||
</view>
|
||||
<view class="unit">m³</view>
|
||||
</view>
|
||||
<!-- 长宽高输入框 -->
|
||||
<view class="long-width-height" v-if="isVolumeInfo">
|
||||
<view class="long">
|
||||
<input class="uni-input" type="digit" :value="long" maxlength="3" placeholder="长 cm"
|
||||
@input="longFun" />
|
||||
</view>
|
||||
<view class="cheng">*</view>
|
||||
<view class="width">
|
||||
<input class="uni-input" type="digit" :value="width" maxlength="3" placeholder="宽 cm"
|
||||
@input="widthFun" />
|
||||
</view>
|
||||
<view class="cheng">*</view>
|
||||
<view class="height">
|
||||
<input class="uni-input" type="digit" :value="height" maxlength="3" placeholder="高 cm"
|
||||
@input="heightFun" />
|
||||
</view>
|
||||
</view>
|
||||
<view class="accounting-rules">
|
||||
注:实际质量与体积以收派员确定为准,物品在包装后重量可能会增加,重量体积大时会体积重量收费,
|
||||
<text class="red" @click="openAccountRulesDialog">了解计费规则</text>
|
||||
</view>
|
||||
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
onMounted,
|
||||
} from 'vue';
|
||||
import {
|
||||
useStore
|
||||
} from 'vuex';
|
||||
const emits = defineEmits(["@getWeight", '@getVolume']);
|
||||
let isLessThan = ref(true); //判断重量是否小于0.1
|
||||
let isExceed = ref(false); //判断重量是否大于9999
|
||||
let isLessThanVolume = ref(true); //判断体积是否小于0.0001m³
|
||||
let isExceedVolume = ref(false); //判断体积是否大于99m³
|
||||
const weight = ref(1) //重量
|
||||
const volume = ref(0) //体积
|
||||
const long = ref() //长
|
||||
const width = ref() //宽
|
||||
const height = ref() //高
|
||||
const isVolumeInfo = ref(false)
|
||||
const store = useStore(); //vuex获取、储存数据
|
||||
const users = store.state.user
|
||||
onMounted(() => {
|
||||
weight.value = users.weight
|
||||
width.value = users.width
|
||||
height.value = users.height
|
||||
long.value = users.long
|
||||
volume.value = users.volume ? (Number(users.volume) / 1000000) : 0
|
||||
isVolumeInfo.value = users.width && users.long && users.height
|
||||
})
|
||||
// 减重量
|
||||
const handleMinus = () => {
|
||||
// 重量减去1
|
||||
if (weight.value > 1) {
|
||||
weight.value--;
|
||||
isExceed.value = false; //右侧加号去除置灰
|
||||
weight.value = weight.value.toFixed(1);
|
||||
}
|
||||
if (weight.value <= 1) {
|
||||
|
||||
isLessThan.value = true; //左侧减号置灰
|
||||
if (weight.value <= 0.1) {
|
||||
weight.value = 0.1
|
||||
}
|
||||
}
|
||||
emits('getWeight', weight.value)
|
||||
};
|
||||
|
||||
// 加重量
|
||||
const handleAdd = () => {
|
||||
// 重量加1
|
||||
if (weight.value < 9999) {
|
||||
++weight.value;
|
||||
isLessThan.value = false; //左侧减号去除置灰
|
||||
}
|
||||
if (weight.value === 9999) {
|
||||
isExceed.value = true; //右侧加号置灰
|
||||
}
|
||||
if (weight.value <= 1) {
|
||||
// weight.value = 1;
|
||||
isLessThan.value = true; //左侧减号置灰
|
||||
}
|
||||
emits('getWeight', weight.value)
|
||||
};
|
||||
|
||||
//触发重量输入如果输入0,自动判断为1kg,最小可输入值为0.1kg,最大值为9999kg
|
||||
const handleWeigthBlur = e => {
|
||||
let value = e.detail.value;
|
||||
if (value < 0.1) {
|
||||
value = 1;
|
||||
isLessThan.value = true; //左侧减号置灰
|
||||
} else {
|
||||
if (value >= 0.1 && value <= 1) {
|
||||
isLessThan.value = true;
|
||||
} else {
|
||||
isLessThan.value = false; //左侧减号去除置灰
|
||||
}
|
||||
|
||||
if (value >= 9999) {
|
||||
isExceed.value = true; //右侧加号置灰
|
||||
value = 9999;
|
||||
uni.showToast({
|
||||
title: '重量最大可不能超过9999kg',
|
||||
duration: 1000,
|
||||
icon: 'none'
|
||||
});
|
||||
} else {
|
||||
isExceed.value = false; //右侧加号去除置灰
|
||||
}
|
||||
}
|
||||
weight.value = value
|
||||
emits('getWeight', weight.value)
|
||||
};
|
||||
|
||||
|
||||
// 体积
|
||||
const handleVolume = e => {
|
||||
let value = Number(e.detail.value);
|
||||
if (value < 0.0001) {
|
||||
isLessThanVolume.value = true;
|
||||
value = 0;
|
||||
} else {
|
||||
isLessThanVolume.value = false;
|
||||
if (value >= 999) {
|
||||
isExceedVolume.value = true; //右侧加号置灰
|
||||
value = 999;
|
||||
uni.showToast({
|
||||
title: '体积最大可不能超过999m³',
|
||||
duration: 1000,
|
||||
icon: 'none'
|
||||
});
|
||||
} else {
|
||||
value = Number(e.detail.value);
|
||||
isExceedVolume.value = false; //右侧加号去除置灰
|
||||
}
|
||||
|
||||
}
|
||||
volume.value = value
|
||||
emits('getVolume', (volume.value) * 1000000, long.value, width.value, height.value)
|
||||
};
|
||||
// 减体积
|
||||
const handleVolumeMinus = () => {
|
||||
// 体积减去1
|
||||
if (volume.value > 1) {
|
||||
volume.value--;
|
||||
isExceedVolume.value = false; //右侧加号去除置灰
|
||||
volume.value = volume.value.toFixed(1);
|
||||
}
|
||||
// 体积
|
||||
if (volume.value <= 1) {
|
||||
|
||||
isLessThanVolume.value = true; //左侧减号置灰
|
||||
if (weight.value <= 0.0001) {
|
||||
weight.value = 0.0001
|
||||
}
|
||||
}
|
||||
emits('getVolume', (volume.value) * 1000000, long.value, width.value, height.value)
|
||||
|
||||
};
|
||||
// 加体积
|
||||
const handleVolumeAdd = () => {
|
||||
// 体积加1
|
||||
if (volume.value < 999) {
|
||||
++volume.value;
|
||||
isLessThanVolume.value = false; //左侧减号去除置灰
|
||||
}
|
||||
if (volume.value === 999) {
|
||||
isExceedVolume.value = true; //右侧加号置灰
|
||||
}
|
||||
emits('getVolume', (volume.value) * 1000000, long.value, width.value, height.value)
|
||||
};
|
||||
|
||||
|
||||
//是否显示具体体积
|
||||
const switchChange = (e) => {
|
||||
isVolumeInfo.value = e.detail.value
|
||||
}
|
||||
//修改长度
|
||||
const longFun = (e) => {
|
||||
long.value = e.detail.value
|
||||
let valueFun = (Number(long.value) * Number(width.value) * Number(height.value)) / 1000000
|
||||
volume.value = valueFun < 0.0001 ? 0.0001 : valueFun
|
||||
emits('getVolume', ((volume.value) * 1000000), e.detail.value, width.value, height.value)
|
||||
}
|
||||
//修改宽度
|
||||
const widthFun = (e) => {
|
||||
width.value = e.detail.value
|
||||
let valueFun = (Number(long.value) * Number(width.value) * Number(height.value)) / 1000000
|
||||
volume.value = valueFun < 0.0001 ? 0.0001 : valueFun
|
||||
emits('getVolume', (volume.value) * 1000000, long.value, e.detail.value, height.value)
|
||||
}
|
||||
//修改高度
|
||||
const heightFun = (e) => {
|
||||
height.value = e.detail.value
|
||||
let valueFun = (Number(long.value) * Number(width.value) * Number(height.value)) / 1000000
|
||||
volume.value = valueFun < 0.0001 ? 0.0001 : valueFun
|
||||
emits('getVolume', (volume.value) * 1000000, long.value, width.value, e.detail.value)
|
||||
}
|
||||
|
||||
//打开计费规则弹窗
|
||||
const openAccountRulesDialog = () => {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/account-rules/index'
|
||||
});
|
||||
}
|
||||
</script>
|
||||
|
||||
<style src="./weightAndVolume.scss" lang="scss" scoped></style>
|
37
project-wl-yonghuduan-uniapp-vue3/pages/goodsInfo/index.scss
Normal file
37
project-wl-yonghuduan-uniapp-vue3/pages/goodsInfo/index.scss
Normal file
@@ -0,0 +1,37 @@
|
||||
.goodsInfo-box{
|
||||
height: 100vh;
|
||||
background-color:#F3F5F9
|
||||
}
|
||||
|
||||
.goods-info{
|
||||
background-color:#F3F5F9;
|
||||
overflow: scroll;
|
||||
.footer{
|
||||
height: 164rpx;
|
||||
width: 100%;
|
||||
background-color: white;
|
||||
position: fixed;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
z-index: 9;
|
||||
.btn{
|
||||
width: 404rpx;
|
||||
height: 88rpx;
|
||||
background-color: #CCCCCC;
|
||||
border-radius: 44rpx;
|
||||
text-align: center;
|
||||
line-height: 88rpx;
|
||||
color: white;
|
||||
margin: 0 auto;
|
||||
margin-top: 17rpx;
|
||||
}
|
||||
.btn.active{
|
||||
background-color:#E84134
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.isAll{
|
||||
background-color: white;
|
||||
height: 100vh;
|
||||
}
|
87
project-wl-yonghuduan-uniapp-vue3/pages/goodsInfo/index.vue
Normal file
87
project-wl-yonghuduan-uniapp-vue3/pages/goodsInfo/index.vue
Normal file
@@ -0,0 +1,87 @@
|
||||
<template>
|
||||
<view class="goodsInfo-box">
|
||||
<!-- 头部导航栏 -->
|
||||
<nav-bar title='物品信息'></nav-bar>
|
||||
<view class="goods-info" :class="{isAll:isShow}">
|
||||
<!-- 物品名称搜索组件 -->
|
||||
<GoodsSeach :isShowOther="isShowOther" @getGoodsInfo="getGoodsInfo"></GoodsSeach>
|
||||
<!-- 预估重量 -->
|
||||
<WeightAndVolume v-if="!isShow" @getWeight="getWeight" @getVolume="getVolume"></WeightAndVolume>
|
||||
<!-- 底部 -->
|
||||
<view class="footer" v-if="!isShow">
|
||||
<view class="btn" :class="isActive?'active':''" @click="confirm">确定</view>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
reactive,
|
||||
onMounted,
|
||||
computed
|
||||
} from 'vue';
|
||||
import {
|
||||
useStore
|
||||
} from 'vuex';
|
||||
import GoodsSeach from './components/goodsSearch.vue'
|
||||
import WeightAndVolume from './components/weightAndVolume.vue'
|
||||
const weight = ref(1)
|
||||
const volume = ref('')
|
||||
const goods = reactive({
|
||||
info: {}
|
||||
})
|
||||
const store = useStore(); //vuex获取、储存数据
|
||||
const users = store.state.user
|
||||
const isShow = ref(false)
|
||||
let isAlways = ref()
|
||||
onMounted(() => {
|
||||
goods.info = users.goodsInfo
|
||||
})
|
||||
const isActive = computed(() => {
|
||||
return Boolean(users.goodsInfo.name)
|
||||
})
|
||||
const isShowOther = (flag, type) => {
|
||||
if (type === 'always') {
|
||||
isAlways.value = true
|
||||
}
|
||||
isShow.value = type === 'always' ? true : flag
|
||||
}
|
||||
//子组件传出来填写的重量的值
|
||||
const getWeight = (value, ) => {
|
||||
weight.value = value
|
||||
store.commit('user/setWeight', value)
|
||||
}
|
||||
//子组件传出来填写的体积的值
|
||||
const getVolume = (value, long, width, height) => {
|
||||
volume.value = value
|
||||
store.commit('user/setLong', long)
|
||||
store.commit('user/setWidth', width)
|
||||
store.commit('user/setHeight', height)
|
||||
store.commit('user/setVolume', value)
|
||||
}
|
||||
//子组件传出来填写的物品的值
|
||||
const getGoodsInfo = (value) => {
|
||||
goods.info = value
|
||||
store.commit('user/setGoodsInfo', value)
|
||||
}
|
||||
|
||||
const confirm = () => {
|
||||
if (!goods.info.name) {
|
||||
return uni.showToast({
|
||||
title: '请选择物品',
|
||||
duration: 1000,
|
||||
icon: 'none'
|
||||
})
|
||||
}
|
||||
uni.redirectTo({
|
||||
url: '/pages/express-delivery/index?isFromGoods=true'
|
||||
});
|
||||
}
|
||||
</script>
|
||||
|
||||
<style src="./components/weightAndVolume.scss" lang="scss"></style>
|
||||
<style lang="scss" src='./components/goodsSearch.scss'></style>
|
||||
<style lang="scss" src='./index.scss'></style>
|
@@ -0,0 +1,196 @@
|
||||
.maxHeight{
|
||||
height: 1170rpx;
|
||||
}
|
||||
.orderList{
|
||||
margin: 40rpx 30rpx 0;
|
||||
position: relative;
|
||||
bottom: 49rpx;
|
||||
::v-deep .scrollView{
|
||||
padding-bottom: 40rpx;
|
||||
}
|
||||
|
||||
//隐藏滚动条
|
||||
::v-deep ::-webkit-scrollbar {
|
||||
display: none;
|
||||
}
|
||||
.title{
|
||||
font-size: 30rpx;
|
||||
color:#333334 ;
|
||||
font-weight: bold;
|
||||
margin-bottom: 10rpx;
|
||||
}
|
||||
.see-more{
|
||||
font-size: 24rpx;
|
||||
text-align: center;
|
||||
margin-top: 22rpx;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
.arrow{
|
||||
margin-left: 10rpx;
|
||||
margin-top: 4rpx;
|
||||
width: 12rpx;
|
||||
height: 18rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url('../../../static/icon15.png');
|
||||
}
|
||||
}
|
||||
.empty-data{
|
||||
.empty-data-image{
|
||||
width: 400rpx;
|
||||
height: 240rpx;
|
||||
text-align: center;
|
||||
margin: 0 auto;
|
||||
display: block;
|
||||
}
|
||||
.tips{
|
||||
color: #818181;
|
||||
font-size: 28rpx;
|
||||
text-align: center;
|
||||
margin-top: 40rpx;
|
||||
margin-left: 25rpx;
|
||||
}
|
||||
}
|
||||
.notLogin{
|
||||
margin-top: 150rpx;
|
||||
.toLogin-btn{
|
||||
width: 240rpx;
|
||||
height: 88rpx;
|
||||
background: #E63E32;
|
||||
border-radius: 44rpx;
|
||||
color: white;
|
||||
font-size: 30rpx;
|
||||
margin: 0 auto;
|
||||
text-align: center;
|
||||
line-height: 88rpx;
|
||||
}
|
||||
.tips{
|
||||
color:#6C6C6C ;
|
||||
font-size: 24rpx;
|
||||
text-align: center;
|
||||
margin-top: 30rpx;
|
||||
}
|
||||
}
|
||||
.order-item{
|
||||
position: relative;
|
||||
background: #FFFFFF;
|
||||
border-radius: 24rpx;
|
||||
padding:28rpx 32rpx 20rpx;
|
||||
margin-top: 20rpx;
|
||||
.orderNumber{
|
||||
font-size: 20rpx;
|
||||
color: #878787;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
.copy{
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
margin-left: 10rpx;
|
||||
}
|
||||
}
|
||||
.area-to-area{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
margin-top:40rpx ;
|
||||
margin-left: 42rpx;
|
||||
margin-right: 42rpx;
|
||||
margin-bottom: 20rpx;
|
||||
.sendBox,.getBox{
|
||||
text-align: center;
|
||||
.sendArea,.getArea{
|
||||
font-size: 30rpx;
|
||||
font-weight: bold;
|
||||
}
|
||||
.sendName,.getName{
|
||||
font-size: 24rpx;
|
||||
color: #878787;
|
||||
}
|
||||
}
|
||||
.order-status{
|
||||
.status{
|
||||
font-size: 24rpx;
|
||||
color: #878787;
|
||||
padding-left: 20rpx;
|
||||
}
|
||||
.arrow{
|
||||
width: 160rpx;
|
||||
height: 20rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
}
|
||||
.green-arrow{
|
||||
background-image: url();
|
||||
}
|
||||
.red-arrow{
|
||||
background-image: url();
|
||||
}
|
||||
.gray-arrow{
|
||||
background-image: url('');
|
||||
}
|
||||
}
|
||||
}
|
||||
.order-detail{
|
||||
margin-top: 12rpx;
|
||||
overflow: hidden;
|
||||
border-bottom: 2rpx solid #F4F4F4;
|
||||
padding-bottom: 26rpx;
|
||||
.detail-item{
|
||||
overflow: hidden;
|
||||
text-overflow:ellipsis;
|
||||
white-space: nowrap;
|
||||
font-size: 24rpx;
|
||||
margin-top: 12rpx;
|
||||
uni-text,text{
|
||||
color: #878787;
|
||||
display: inline-block;
|
||||
}
|
||||
}
|
||||
}
|
||||
.btn-box{
|
||||
.btn{
|
||||
width: 144rpx;
|
||||
height: 56rpx;
|
||||
background: #FFFFFF;
|
||||
border: 2rpx solid #DDDDDD;
|
||||
border-radius: 28rpx;
|
||||
text-align: center;
|
||||
line-height: 56rpx;
|
||||
font-size: 24rpx;
|
||||
color:#303030 ;
|
||||
}
|
||||
.wait-pay{
|
||||
margin-top: 29rpx;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
|
||||
justify-content: space-between;
|
||||
.price{
|
||||
font-size: 26rpx;
|
||||
font-weight: bold;
|
||||
uni-text,text{
|
||||
color: #E84134;
|
||||
}
|
||||
}
|
||||
}
|
||||
.btn-list{
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
margin-top: 19rpx;
|
||||
.btn{
|
||||
margin-left: 20rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
.paymentStatus{
|
||||
position: absolute;
|
||||
right: 29rpx;
|
||||
top: 27rpx;
|
||||
image{
|
||||
width: 76rpx;
|
||||
height: 30rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,281 @@
|
||||
<template>
|
||||
<view class="orderList">
|
||||
<view v-if="!isLogin" class="notLogin">
|
||||
<view class="toLogin-btn" @click="toLogin">登录/注册</view>
|
||||
<view class="tips">登录后可查看快递信息</view>
|
||||
</view>
|
||||
|
||||
|
||||
<view v-else-if="!allOrderList.data.length && isLogin" class="empty-data">
|
||||
<image src="../../../static/emptyData.png" class="empty-data-image"></image>
|
||||
<view class="tips">没有运单~</view>
|
||||
</view>
|
||||
<view v-else>
|
||||
<view class="title">运单信息</view>
|
||||
|
||||
<view class="order-item" v-for="(item,index) in allOrderList.data" :key="index"
|
||||
@click='handleToOrderInfo(event,item.id,item.transportOrderId)'>
|
||||
<view class="orderNumber">
|
||||
{{[23000,22000,230011].includes(item.status)?'订':'运'}}单号:{{[23000,22000,230011].includes(item.status)?item.id:item.transportOrderId}}
|
||||
<image src="../../../static/pickUp-copy.png" class="copy"
|
||||
@click.stop="handleCopy([23000,22000,230011].includes(item.status)?item.id:item.transportOrderId)">
|
||||
</image>
|
||||
</view>
|
||||
<view class="area-to-area">
|
||||
<view class="sendBox">
|
||||
<view class="sendArea">{{item.senderCity.name}}</view>
|
||||
<view class="sendName">{{item.senderName}}</view>
|
||||
</view>
|
||||
<view class="order-status">
|
||||
<view class="status">
|
||||
{{showOrderStatus(item.status)}}
|
||||
</view>
|
||||
<view class="arrow" :class="
|
||||
[[21000,23000,23001,23005,23008].includes(item.status)?'green-arrow':'',
|
||||
[23009,23010].includes(item.status)?'red-arrow':'',
|
||||
[230011,22000].includes(item.status)?'gray-arrow':'']
|
||||
"></view>
|
||||
</view>
|
||||
<view class="getBox">
|
||||
<view class="getArea">{{item.receiverCity.name}}</view>
|
||||
<view class="getName">{{item.receiverName}}</view>
|
||||
|
||||
</view>
|
||||
|
||||
</view>
|
||||
<view class="order-detail">
|
||||
<!-- 已取件运输信息 -->
|
||||
<view v-if="item.status ===23001 && item.transportOrderPointVOS" class="detail-item">
|
||||
<text>已取件:</text>
|
||||
{{ item.transportOrderPointVOS.length>0?item.transportOrderPointVOS[item.transportOrderPointVOS.length-1].info:''}}
|
||||
</view>
|
||||
<!-- 运输中信息 -->
|
||||
<view v-if="[23005,23008].includes(item.status) && item.transportOrderPointVOS" class="detail-item">
|
||||
<text>运送中:</text>{{ item.transportOrderPointVOS.length>0?item.transportOrderPointVOS[item.transportOrderPointVOS.length-1].info:''}}
|
||||
</view>
|
||||
<!-- 已签收 -->
|
||||
<view v-if="item.status ===23009 && item.transportOrderPointVOS" class="detail-item">
|
||||
<text>已签收:</text>{{ item.transportOrderPointVOS.length>0?item.transportOrderPointVOS[item.transportOrderPointVOS.length-1].info:''}}
|
||||
</view>
|
||||
<!-- 已拒收 -->
|
||||
<view v-if="item.status ===23010 && item.transportOrderPointVOS" class="detail-item">
|
||||
<text>已拒收:</text>{{ item.transportOrderPointVOS.length>0?item.transportOrderPointVOS[item.transportOrderPointVOS.length-1].info:''}}
|
||||
</view>
|
||||
<!-- 预计上门时间 -->
|
||||
<view v-if="item.status ===23000 " class="detail-item">
|
||||
<text>预计上门时间:</text>{{item.estimatedStartTime}}
|
||||
</view>
|
||||
<!-- 取消时间 -->
|
||||
<view v-if="item.status ===230011 " class="detail-item"><text>取消时间:</text>{{item.updated}}</view>
|
||||
<!-- 预计送达时间 -->
|
||||
<view v-if="[23001,23005,23008,23010].includes(item.status) " class="detail-item">
|
||||
<text>预计送达时间:</text>{{item.estimatedArrivalTime}}
|
||||
</view>
|
||||
<!-- 已关闭时间 -->
|
||||
<view v-if="item.status ===22000 " class="detail-item"><text>关闭时间:</text>{{item.updated}}</view>
|
||||
<!-- 签收时间 -->
|
||||
<view v-if="item.status ===23009 " class="detail-item"><text>签收时间:</text>{{item.updated}}</view>
|
||||
</view>
|
||||
<!-- 功能按钮区域 -->
|
||||
<view class="btn-box">
|
||||
<view class="btn-list" @click.stop="handleSecondQi">
|
||||
<view class="btn" v-if="[23000,23001,23005,23008,23010].includes(item.status)">分享</view>
|
||||
<view class="btn" v-if='item.status===23000' @click.stop="handleOrderCancel(item.id)">取消寄件
|
||||
</view>
|
||||
<view class="btn" v-if="[22000,230011,23009].includes(item.status)"
|
||||
@click.stop="handleOrderDelete(item.id)">删除</view>
|
||||
</view>
|
||||
</view>
|
||||
<view class="paymentStatus"
|
||||
v-if="item.paymentStatus && [23001,23005,23008,23009,23010].includes(item.status)">
|
||||
<image :src="item.paymentStatus===1?'../../../static/daizhifu.png':'../../../static/yizhifu.png'">
|
||||
</image>
|
||||
</view>
|
||||
</view>
|
||||
<view class="see-more" @click="handleSeeMore">
|
||||
查看更多
|
||||
<view class="arrow"></view>
|
||||
</view>
|
||||
|
||||
</view>
|
||||
|
||||
<!-- 删除确认对话框 -->
|
||||
<uni-popup ref="popup" type="dialog" class='address-popup'>
|
||||
<uni-popup-dialog mode="base" :content="'确定是否删除此条订单?'" :animation='false' :before-close="true"
|
||||
@close="close" @confirm="confirm">
|
||||
|
||||
</uni-popup-dialog>
|
||||
</uni-popup>
|
||||
</view>
|
||||
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
reactive,
|
||||
onMounted,
|
||||
} from 'vue';
|
||||
import {
|
||||
getOrderList,
|
||||
deleteOrder
|
||||
} from '@/pages/api/order.js'
|
||||
import {
|
||||
onShow,
|
||||
} from '@dcloudio/uni-app';
|
||||
import {
|
||||
handleSecondQi
|
||||
} from '@/utils/index.js'
|
||||
import {
|
||||
useStore
|
||||
} from 'vuex';
|
||||
const store = useStore(); //vuex获取、储存数据
|
||||
const emits = defineEmits(["@stopRefresh"]);
|
||||
let pageInfo = reactive({
|
||||
page: 1,
|
||||
pageSize: 10
|
||||
})
|
||||
let status = ref('more') //加载状态
|
||||
let scrollTop = ref(0) //顶部位置
|
||||
let allOrderList = reactive({
|
||||
data: []
|
||||
})
|
||||
let orderId = ref()
|
||||
let isLogin = ref('')
|
||||
let popup = ref(null)
|
||||
//关闭删除确认提示框
|
||||
const close = () => {
|
||||
popup.value.close()
|
||||
orderId.value = ''
|
||||
}
|
||||
onShow((options) => {
|
||||
isLogin.value = uni.getStorageSync('token')
|
||||
console.log(!isLogin, (!allOrderList.data.length && isLogin.value), '6666')
|
||||
})
|
||||
onMounted(() => {
|
||||
getOrderListFunc()
|
||||
})
|
||||
//复制
|
||||
const handleCopy = (value) => {
|
||||
uni.setClipboardData({
|
||||
data: value,
|
||||
showToast: false,
|
||||
success: () => {
|
||||
uni.hideToast(); // 隐藏弹出提示
|
||||
uni.hideKeyboard(); // 隐藏软键盘
|
||||
uni.showToast({
|
||||
title: '复制成功',
|
||||
icon: 'success',
|
||||
duration: 1000
|
||||
})
|
||||
|
||||
}
|
||||
});
|
||||
}
|
||||
//确认删除订单
|
||||
const confirm = () => {
|
||||
popup.value.close()
|
||||
deleteOrder(orderId.value).then((res) => {
|
||||
pageInfo.page = 1
|
||||
pageInfo.pageSize = 10
|
||||
getOrderListFunc()
|
||||
uni.showToast({
|
||||
title: '删除成功',
|
||||
icon: 'success',
|
||||
duration: 1000
|
||||
})
|
||||
}).catch((err) => {
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
});
|
||||
})
|
||||
}
|
||||
//删除订单
|
||||
const handleOrderDelete = (id) => {
|
||||
orderId.value = id
|
||||
popup.value.open()
|
||||
}
|
||||
//跳转到取消订单页面
|
||||
const handleOrderCancel = (id) => {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/order-cancel/index?orderId=' + id
|
||||
})
|
||||
}
|
||||
const toLogin = () => {
|
||||
uni.navigateTo({
|
||||
url: '/pages/login/index'
|
||||
})
|
||||
}
|
||||
const getOrderListFunc = (flag) => {
|
||||
status.value = 'loading'
|
||||
getOrderList({
|
||||
page: pageInfo.page,
|
||||
pageSize: pageInfo.pageSize,
|
||||
}).then((res) => {
|
||||
console.log(res, 'getOrderListFunc')
|
||||
if (res.data) {
|
||||
allOrderList.data = res.data.items ? res.data.items.slice(0, 3) : []
|
||||
}
|
||||
})
|
||||
}
|
||||
//查看更多
|
||||
const handleSeeMore = () => {
|
||||
uni.switchTab({
|
||||
url: '/pages/pickup/index'
|
||||
})
|
||||
}
|
||||
|
||||
//根据状态去显示对应的运单状态文案
|
||||
const showOrderStatus = (status) => {
|
||||
switch (status) {
|
||||
case 21000:
|
||||
return '待支付';
|
||||
case 23000:
|
||||
return '待取件';
|
||||
case 230011:
|
||||
return '已取消';
|
||||
case 23001:
|
||||
return '已取件';
|
||||
case 23005:
|
||||
return '运送中';
|
||||
case 22000:
|
||||
return '已关闭';
|
||||
case 23008:
|
||||
return '派送中';
|
||||
case 23009:
|
||||
return '已签收';
|
||||
case 23010:
|
||||
return '已拒收';
|
||||
}
|
||||
}
|
||||
//跳转到订单详情页面
|
||||
const handleToOrderInfo = (event, id, transportOrderId) => {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/order-info/index?orderId=' + id + '&transportOrderId=' + transportOrderId
|
||||
});
|
||||
store.commit('user/setIsToOrderInfo', true)
|
||||
}
|
||||
|
||||
//下拉加载更多
|
||||
const LoadMoreCustomers = () => {
|
||||
if (status.value === 'no-more') {
|
||||
return
|
||||
}
|
||||
pageInfo.page = pageInfo.page + 1
|
||||
getOrderListFunc('topPull')
|
||||
}
|
||||
//
|
||||
const indexGetOrderListFunc = () => {
|
||||
pageInfo.page = 1
|
||||
getOrderListFunc()
|
||||
}
|
||||
// 暴漏给父组件
|
||||
defineExpose({
|
||||
indexGetOrderListFunc,
|
||||
LoadMoreCustomers
|
||||
});
|
||||
</script>
|
||||
|
||||
<style src="./orderList.scss" lang="scss" scoped></style>
|
68
project-wl-yonghuduan-uniapp-vue3/pages/index/index.scss
Normal file
68
project-wl-yonghuduan-uniapp-vue3/pages/index/index.scss
Normal file
@@ -0,0 +1,68 @@
|
||||
@import url('@/styles/theme.scss');
|
||||
.scrollView{
|
||||
background-color: #F3F5F9 !important;
|
||||
height: 100vh;
|
||||
}
|
||||
.homePage{
|
||||
background-color: #F3F5F9 !important;
|
||||
position: relative;
|
||||
// padding-bottom: 30rpx;
|
||||
min-height: 100vh;
|
||||
image{
|
||||
width: 100%;
|
||||
height: 400rpx;
|
||||
}
|
||||
.feature-top{
|
||||
height: 80rpx;
|
||||
margin: 0 28rpx;
|
||||
background: #FFFFFF;
|
||||
border-radius: 24rpx;
|
||||
display: flex;
|
||||
padding: 48rpx;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
position: relative;
|
||||
bottom: 48rpx;
|
||||
.feature-top-line{
|
||||
position: absolute;
|
||||
left: 50%;
|
||||
height: 114rpx;
|
||||
background-color:#F4F4F4 ;
|
||||
width: 2rpx;
|
||||
}
|
||||
}
|
||||
.jikuaidi,.saomaji,.piliangji,.saomaji{
|
||||
display: flex;
|
||||
image{
|
||||
width: 80rpx;
|
||||
height: 80rpx;
|
||||
margin-right: 15rpx;
|
||||
}
|
||||
.des-title{
|
||||
font-size: 30rpx;
|
||||
font-weight: bold;
|
||||
color: #151515;
|
||||
}
|
||||
.des-dec{
|
||||
font-size: 22rpx;
|
||||
color: #878787;
|
||||
}
|
||||
}
|
||||
.feature-bottom{
|
||||
display: flex;
|
||||
padding: 20rpx 32rpx;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
position: relative;
|
||||
bottom: 49rpx;
|
||||
.piliangji,.saomaji{
|
||||
height:128rpx ;
|
||||
width: 334rpx;
|
||||
background: #FFFFFF;
|
||||
border-radius: 24rpx;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
}
|
||||
}
|
||||
}
|
91
project-wl-yonghuduan-uniapp-vue3/pages/index/index.vue
Normal file
91
project-wl-yonghuduan-uniapp-vue3/pages/index/index.vue
Normal file
@@ -0,0 +1,91 @@
|
||||
<!-- 首页 -->
|
||||
<template>
|
||||
<view class="homePage">
|
||||
<!-- banner图 -->
|
||||
<image src='../../static/tupian-banner.png' />
|
||||
<!-- 功能列表 -->
|
||||
<!-- 寄快递和扫码寄 -->
|
||||
<view class="feature-top">
|
||||
<view class="jikuaidi" @click="toExpressDelivery">
|
||||
<image src='../../static/fe-jikuaidi.png' />
|
||||
<view class="des">
|
||||
<view class="des-title">寄快递</view>
|
||||
<view class="des-dec">1小时上门取件</view>
|
||||
</view>
|
||||
</view>
|
||||
<view class="feature-top-line"></view>
|
||||
<view class="saomaji" @click="handleSecondQi">
|
||||
<image src='../../static/fe-saomaji.png' />
|
||||
<view class="des">
|
||||
<view class="des-title">扫码寄</view>
|
||||
<view class="des-dec">扫二维码下单</view>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 批量寄和礼物寄 -->
|
||||
<view class="feature-bottom">
|
||||
<view class="piliangji" @click="handleSecondQi">
|
||||
<image src='../../static/fe-piliangji.png' />
|
||||
<view class="des">
|
||||
<view class="des-title">批量寄</view>
|
||||
<view class="des-dec">便捷寄多个快递</view>
|
||||
</view>
|
||||
</view>
|
||||
<view class="saomaji" @click="handleSecondQi">
|
||||
<image src='../../static/fe-liwuji.png' />
|
||||
<view class="des">
|
||||
<view class="des-title">礼物寄</view>
|
||||
<view class="des-dec">保留神秘寄</view>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 运单列表 -->
|
||||
<OrderList ref='orderListRef' @stopRefresh="stopRefreshFunc"></OrderList>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref
|
||||
} from 'vue';
|
||||
import {
|
||||
onShow,
|
||||
onPullDownRefresh,
|
||||
} from '@dcloudio/uni-app';
|
||||
import {
|
||||
handleSecondQi
|
||||
} from '@/utils/index.js'
|
||||
//运单列表
|
||||
import OrderList from './components/orderList';
|
||||
import {
|
||||
useStore
|
||||
} from 'vuex';
|
||||
const store = useStore(); //vuex获取、储存数据
|
||||
const users = store.state.user
|
||||
let orderListRef = ref()
|
||||
// ------定义变量------
|
||||
onShow(() => {
|
||||
// if (users.isToOrderInfo) {
|
||||
// store.commit('user/setIsToOrderInfo', false)
|
||||
// } else {
|
||||
// orderListRef.value && orderListRef.value.indexGetOrderListFunc()
|
||||
// }
|
||||
orderListRef.value && orderListRef.value.indexGetOrderListFunc()
|
||||
})
|
||||
// ------生命周期------
|
||||
onPullDownRefresh(() => {
|
||||
orderListRef.value.indexGetOrderListFunc()
|
||||
});
|
||||
//
|
||||
const stopRefreshFunc = () => {
|
||||
uni.stopPullDownRefresh();
|
||||
}
|
||||
//跳转到寄快递页面
|
||||
const toExpressDelivery = () => {
|
||||
uni.navigateTo({
|
||||
url: uni.getStorageSync('token') ? '/pages/express-delivery/index' : '/pages/login/index'
|
||||
});
|
||||
}
|
||||
|
||||
</script>
|
||||
<style src="./index.scss" lang="scss" scoped></style>
|
84
project-wl-yonghuduan-uniapp-vue3/pages/login/index.scss
Normal file
84
project-wl-yonghuduan-uniapp-vue3/pages/login/index.scss
Normal file
@@ -0,0 +1,84 @@
|
||||
|
||||
.loginBox {
|
||||
background-color: white;
|
||||
height: 100vh;
|
||||
::v-deep .uni-popup__wrapper {
|
||||
height:500rpx;
|
||||
background-color: white !important;
|
||||
border-radius: 24rpx 24rpx 0 0;
|
||||
padding: 65rpx 39rpx 180rpx;
|
||||
}
|
||||
.logo-box{
|
||||
text-align: center;
|
||||
margin-top: 204rpx;
|
||||
image{
|
||||
width: 166rpx;
|
||||
height: 174rpx;
|
||||
}
|
||||
.title{
|
||||
font-size: 32rpx;
|
||||
color:#20232A ;
|
||||
font-weight: bold;
|
||||
|
||||
}
|
||||
}
|
||||
.open-dialog{
|
||||
height: 88rpx;
|
||||
background: #E63E32;
|
||||
border-radius: 44rpx;
|
||||
color: white;
|
||||
text-align: center;
|
||||
line-height: 88rpx;
|
||||
margin: 228rpx 46rpx 0;
|
||||
}
|
||||
|
||||
uni-popup{
|
||||
.header{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: flex-end;
|
||||
border-radius: 24rpx 24rpx 0 0;
|
||||
image{
|
||||
height: 35rpx;
|
||||
width:35rpx ;
|
||||
}
|
||||
.title{
|
||||
font-weight: bold;
|
||||
font-size: 30rpx;
|
||||
margin: 0 auto;
|
||||
}
|
||||
}
|
||||
.content{
|
||||
font-size: 28rpx;
|
||||
margin-top: 35rpx;
|
||||
.tips{
|
||||
margin-top: 50rpx;
|
||||
}
|
||||
text{
|
||||
font-weight: bold;
|
||||
}
|
||||
}
|
||||
.footer{
|
||||
display: flex;
|
||||
margin-top: 90rpx;
|
||||
.btn{
|
||||
width: 276rpx;
|
||||
height: 88rpx;
|
||||
line-height: 88rpx;
|
||||
border: 2rpx solid #E84134;
|
||||
border-radius: 44rpx;
|
||||
font-size: 30rpx;
|
||||
}
|
||||
.cancel-btn{
|
||||
background: #FFFFFF;
|
||||
color: #E84134;
|
||||
}
|
||||
.agree-btn{
|
||||
background: #E84134;
|
||||
color: white;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
106
project-wl-yonghuduan-uniapp-vue3/pages/login/index.vue
Normal file
106
project-wl-yonghuduan-uniapp-vue3/pages/login/index.vue
Normal file
@@ -0,0 +1,106 @@
|
||||
<!-- 手机短信登录页 -->
|
||||
<template>
|
||||
<view class="loginBox">
|
||||
<!-- 头部导航栏 -->
|
||||
<NavBar title='登录'></NavBar>
|
||||
<view class="logo-box">
|
||||
<image src='../../static/userLogo.png'></image>
|
||||
</view>
|
||||
|
||||
<view class="open-dialog" @click="handleOpen">微信登录</view>
|
||||
|
||||
<!-- 服务条款及隐私政策弹窗 -->
|
||||
<uni-popup ref="popup" type="bottom" :safe-area="false">
|
||||
<view class="header">
|
||||
<view class="title">服务条款及隐私政策</view>
|
||||
<image src='../../static/guanbi.png' @click="handleClose"></image>
|
||||
</view>
|
||||
<view class="content">
|
||||
<view>
|
||||
在您注册成为神领快递会员的过程中您需要通过点击同意的形式在线签署<text>《神领快递服务条款》</text>、<text>《神领快递隐私政策</text>》请您务必仔细阅读充分理解条款内容后再点击同意尤其是以粗体并下划线标识的条款因为这些条款可能会明确您应履行的义务或对您的权利有所限制。
|
||||
</view>
|
||||
<view class="tips">
|
||||
请您注意:如果您不同意上述服务条款隐私政策或其中任何约定请您停止注册
|
||||
</view>
|
||||
</view>
|
||||
<view class="footer">
|
||||
<button class="cancel-btn btn" @click="handleClose">不同意</button>
|
||||
<button class="agree-btn btn" open-type="getPhoneNumber"
|
||||
@getphonenumber="decryptPhoneNumber">同意</button>
|
||||
</view>
|
||||
</uni-popup>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
} from 'vue';
|
||||
// 接口
|
||||
import {
|
||||
login
|
||||
} from '../api/login.js';
|
||||
import NavBar from '@/components/Navbar/index.vue'
|
||||
import {
|
||||
useStore
|
||||
} from 'vuex';
|
||||
// ------定义变量-----
|
||||
const popup = ref();
|
||||
const store = useStore(); //vuex获取、储存数据
|
||||
const handleOpen = () => {
|
||||
popup.value.open()
|
||||
}
|
||||
// 跳转到首页
|
||||
const decryptPhoneNumber = (e) => {
|
||||
handleClose()
|
||||
|
||||
console.log('fff')
|
||||
wx.login({
|
||||
success(res) {
|
||||
//允许授权
|
||||
if (e.detail.errMsg === 'getPhoneNumber:ok' && e.target.errMsg === 'getPhoneNumber:ok') {
|
||||
if (!store.state.user.isLoginSuccess) return uni.showToast({
|
||||
title: '请勿重复登录',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
});
|
||||
store.commit('user/setIsLoginSuccess', false)
|
||||
login({
|
||||
code: res.code,
|
||||
phoneCode: e.detail.code
|
||||
}).then((res) => {
|
||||
console.log(res,'----------------')
|
||||
//将token存到缓存中,后续在统一请求头上加上token(短令牌)
|
||||
uni.setStorageSync('token', res.data.accessToken);
|
||||
//长令牌
|
||||
uni.setStorageSync('refreshToken', res.data.refreshToken);
|
||||
//登录成功后跳转到首页
|
||||
uni.switchTab({
|
||||
url: '/pages/index/index'
|
||||
});
|
||||
store.commit('user/setIsLoginSuccess', true)
|
||||
}).catch((err) => {
|
||||
console.log(err,'==================')
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
});
|
||||
store.commit('user/setIsLoginSuccess', true)
|
||||
})
|
||||
} else {
|
||||
console.log(err,'++++++++++++')
|
||||
uni.redirectTo({
|
||||
url: '/pages/login/index'
|
||||
});
|
||||
}
|
||||
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
const handleClose = () => {
|
||||
popup.value.close()
|
||||
}
|
||||
</script>
|
||||
<style src="./index.scss" lang="scss" scoped></style>
|
114
project-wl-yonghuduan-uniapp-vue3/pages/my/index.scss
Normal file
114
project-wl-yonghuduan-uniapp-vue3/pages/my/index.scss
Normal file
File diff suppressed because one or more lines are too long
198
project-wl-yonghuduan-uniapp-vue3/pages/my/index.vue
Normal file
198
project-wl-yonghuduan-uniapp-vue3/pages/my/index.vue
Normal file
@@ -0,0 +1,198 @@
|
||||
<!-- 我的页面 -->
|
||||
<template>
|
||||
<view class="my">
|
||||
<!-- banner图 -->
|
||||
<view class="banner">
|
||||
<!-- 头像用户名 -->
|
||||
<view class="headBox" @click="toLogin">
|
||||
<image :src="avatarUrl" />
|
||||
<view class="userName">{{nickName}}</view>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 地址簿,实名认证,专属快递员 -->
|
||||
<view class="list-top">
|
||||
<view class="list-item" @click="handleToAddress">
|
||||
<view class="left">
|
||||
<image class="icon" src='../../static/my-address.png' />
|
||||
<view class="label">地址簿</view>
|
||||
</view>
|
||||
|
||||
<image class="arrow" src='../../static/icon15.png' />
|
||||
</view>
|
||||
<view class="list-item" @click="handleTorealName">
|
||||
<view class="left">
|
||||
<image class="icon" src='../../static/my-name.png' />
|
||||
<view class="label">实名认证</view>
|
||||
</view>
|
||||
|
||||
<image class="arrow" src='../../static/icon15.png' />
|
||||
</view>
|
||||
<view class="list-item" @click="handleSecondQi">
|
||||
<view class="left">
|
||||
<image class="icon" src='../../static/my-send.png' />
|
||||
<view class="label">专属快递员</view>
|
||||
</view>
|
||||
|
||||
<image class="arrow" src='../../static/icon15.png' />
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<view class="list-bottom" v-if="token">
|
||||
<picker mode="selector" @change="changeSex" :value="sex" :range="sexList">
|
||||
<view class="list-item sex">
|
||||
<view class="label">性别</view>
|
||||
<view class="item-value">{{sex?'男':'女'}}</view>
|
||||
<image class="arrow" src='../../static/icon15.png' />
|
||||
</view>
|
||||
</picker>
|
||||
<view class="line"></view>
|
||||
<picker mode="date" :value="birthday" :start="startDate" :end="endDate" @change="getDateChange">
|
||||
<view class="list-item">
|
||||
<view class="label">生日</view>
|
||||
<view class="item-value">{{birthday}}</view>
|
||||
<image class="arrow" src='../../static/icon15.png' />
|
||||
</view>
|
||||
</picker>
|
||||
</view>
|
||||
|
||||
<view class="logout" @click="handleLogout" v-if="token">退出登录</view>
|
||||
</view>
|
||||
<!-- end -->
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
reactive,
|
||||
} from 'vue';
|
||||
import {
|
||||
getUserInfo,
|
||||
updateUserInfo
|
||||
} from '@/pages/api/my.js';
|
||||
import {
|
||||
onShow,
|
||||
} from '@dcloudio/uni-app';
|
||||
import {
|
||||
handleSecondQi
|
||||
} from '@/utils/index.js'
|
||||
const token = ref()
|
||||
const sexList = reactive(['女', '男'])
|
||||
let nickName = ref('')
|
||||
let avatarUrl = ref('')
|
||||
let startDate = ref()
|
||||
let endDate = ref()
|
||||
let idCard = ref('')
|
||||
let name = ref('')
|
||||
let sex = ref(1)
|
||||
let birthday = ref('1995-09-01')
|
||||
const isRealNameAuth = ref(false)
|
||||
onShow(() => {
|
||||
startDate.value = getDate('start')
|
||||
endDate.value = getDate('end')
|
||||
nickName.value = uni.getStorageSync('nickName') || '神领用户'
|
||||
avatarUrl.value = uni.getStorageSync('avatarUrl') || '../../static/defaultHeadImg.png'
|
||||
token.value = uni.getStorageSync('token')
|
||||
baseUserInfo()
|
||||
})
|
||||
//退出登录
|
||||
const handleLogout = () => {
|
||||
uni.removeStorageSync('token');
|
||||
uni.removeStorageSync('nickName');
|
||||
uni.removeStorageSync('avatarUrl');
|
||||
uni.switchTab({
|
||||
url: '/pages/index/index'
|
||||
});
|
||||
}
|
||||
const baseUserInfo = () => {
|
||||
getUserInfo().then((res) => {
|
||||
if (res) {
|
||||
isRealNameAuth.value = Boolean(res.data.idCardNoVerify)
|
||||
idCard.value = res.data.idCardNo || ''
|
||||
name.value = res.data.name
|
||||
sex.value = res.data.sex
|
||||
birthday.value = res.data.birthday
|
||||
nickName.value = res.data.phone
|
||||
uni.setStorageSync('nickName', res.data.phone);
|
||||
}
|
||||
|
||||
})
|
||||
}
|
||||
const getDate = (type) => {
|
||||
const date = new Date()
|
||||
let year = date.getFullYear()
|
||||
let month = date.getMonth() + 1
|
||||
let day = date.getDate()
|
||||
if (type === "start") {
|
||||
year = year - 60
|
||||
} else {
|
||||
year = year + 2
|
||||
}
|
||||
month = month > 9 ? month : "0" + month
|
||||
day = day > 9 ? day : '0' + day
|
||||
return `${year}-${month}-${day}`
|
||||
|
||||
}
|
||||
const toLogin = () => {
|
||||
if (uni.getStorageSync('token')) {
|
||||
getUserInfoFunc()
|
||||
} else {
|
||||
uni.navigateTo({
|
||||
url: '/pages/login/index'
|
||||
});
|
||||
}
|
||||
}
|
||||
//跳转到地址簿
|
||||
const handleToAddress = () => {
|
||||
uni.navigateTo({
|
||||
url: uni.getStorageSync('token') ? '/pages/address/index?type=address&isFromAddress=true' :
|
||||
'/pages/login/index'
|
||||
});
|
||||
}
|
||||
//跳转到实名认证页面
|
||||
const handleTorealName = () => {
|
||||
if (!uni.getStorageSync('token')) {
|
||||
return uni.navigateTo({
|
||||
url: '/pages/login/index'
|
||||
});
|
||||
|
||||
}
|
||||
if (isRealNameAuth.value) {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/authentication-success/index?name=' + name.value + '&idCard=' + idCard.value
|
||||
});
|
||||
} else {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/realName-authentication/index'
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
const getDateChange = (e) => {
|
||||
updateUserInfo({
|
||||
birthday: e.detail.value
|
||||
}).then((res) => {
|
||||
baseUserInfo()
|
||||
}).catch((err) => {
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
});
|
||||
})
|
||||
}
|
||||
|
||||
const changeSex = (e) => {
|
||||
updateUserInfo({
|
||||
sex: Number(e.detail.value)
|
||||
}).then((res) => {
|
||||
baseUserInfo()
|
||||
}).catch((err) => {
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
});
|
||||
})
|
||||
}
|
||||
</script>
|
||||
<style src="./index.scss" lang="scss" scoped></style>
|
@@ -0,0 +1,138 @@
|
||||
<template>
|
||||
<!-- 待取件 -->
|
||||
<DealParcel ref="dealparcel" :tabIndex="tabIndex" :isAdmin="isAdmin" @checkbox="checkbox" @getSelected="getSelected"></DealParcel>
|
||||
<!-- end -->
|
||||
<!-- 已取件 -->
|
||||
<AlreadyParcel ref="already" :tabIndex="tabIndex" :isAdmin="isAdmin" @checkbox="checkbox"></AlreadyParcel>
|
||||
<!-- end -->
|
||||
<!-- 已取件 -->
|
||||
<CancelParcel :tabIndex="tabIndex" ref="cancel" :isAdmin="isAdmin" @checkbox="checkbox" @handleOpen="handleOpen"></CancelParcel>
|
||||
<!-- end -->
|
||||
<!-- 提示窗 -->
|
||||
<UniPopup ref="popup" :tipInfo="tipInfo" @handleClick="handleClick"></UniPopup>
|
||||
<!-- end -->
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, onMounted } from 'vue';
|
||||
import { useStore } from 'vuex';
|
||||
//接口
|
||||
import { taskDelete } from '@/pages/api/index.js';
|
||||
// 导入组件
|
||||
// 待取件
|
||||
import DealParcel from './components/dealParcel.vue';
|
||||
// 已取件
|
||||
import AlreadyParcel from './components/alreadyParcel.vue';
|
||||
// 已取件
|
||||
import CancelParcel from './components/cancelParcel.vue';
|
||||
// 弹层
|
||||
import UniPopup from '@/components/uni-popup/index.vue';
|
||||
|
||||
// 获取父组件数据
|
||||
const props = defineProps({
|
||||
// tab切换数据
|
||||
tabBars: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
tabIndex: {
|
||||
type: Number,
|
||||
default: 0
|
||||
},
|
||||
// 当前高度
|
||||
scrollH: {
|
||||
type: String,
|
||||
default: ''
|
||||
},
|
||||
// 是否触发管理按钮
|
||||
isAdmin: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
// 获取当前筛选的距离升序还是降序
|
||||
orderDistance: {
|
||||
type: Number,
|
||||
default: 0
|
||||
},
|
||||
// 获取当前筛选的时间升序还是降序
|
||||
orderTime: {
|
||||
type: Number,
|
||||
default: 0
|
||||
},
|
||||
// 获取当前筛选超时
|
||||
filterOverTime: {
|
||||
type: Number,
|
||||
default: 0
|
||||
}
|
||||
});
|
||||
// ------定义变量------
|
||||
const emit = defineEmits(''); //子组件向父组件事件传递
|
||||
const store = useStore(); //设置、获取储存的数据
|
||||
const users = store.state.user;
|
||||
let popup = ref();
|
||||
let dealparcel = ref();
|
||||
let already = ref();
|
||||
let cancel = ref();
|
||||
const tipInfo = ref('确定要删除吗?');
|
||||
let taskId = ref('');
|
||||
let scrollH = ref(0); //滚动高度
|
||||
// ------生命周期------
|
||||
onMounted(() => {
|
||||
// 获取屏幕信息
|
||||
uni.getSystemInfo({
|
||||
success: res => {
|
||||
scrollH.value = res.windowHeight;
|
||||
}
|
||||
});
|
||||
});
|
||||
// ------定义方法------
|
||||
// 获取已经选的任务
|
||||
const getSelected = array => {
|
||||
emit('getSelected', array);
|
||||
};
|
||||
// 获取待取件列表方法
|
||||
const dealPList = () => {
|
||||
dealparcel.value.getList();
|
||||
};
|
||||
// 获取已取件列表方法
|
||||
const alreadList = () => {
|
||||
already.value.getList();
|
||||
};
|
||||
// 获取取消件列表方法
|
||||
const cancelList = () => {
|
||||
cancel.value.getList();
|
||||
};
|
||||
// 确认删除
|
||||
const handleClick = async () => {
|
||||
await taskDelete(taskId.value)
|
||||
.then(res => {
|
||||
if (res.code === 200) {
|
||||
dealparcel.value.getList();
|
||||
return uni.showToast({
|
||||
title: '删除成功!',
|
||||
duration: 1000,
|
||||
icon: 'none'
|
||||
});
|
||||
}
|
||||
})
|
||||
.catch(err => {});
|
||||
};
|
||||
// 选项框点击事件,参数是数据的下标
|
||||
const checkbox = index => {
|
||||
emit('checkbox', index);
|
||||
};
|
||||
// 删除弹层id
|
||||
const handleOpen = id => {
|
||||
popup.value.dialogOpen();
|
||||
taskId.value = id;
|
||||
};
|
||||
|
||||
//把数据、方法暴漏给父组件
|
||||
defineExpose({
|
||||
dealPList,
|
||||
alreadList,
|
||||
cancelList
|
||||
});
|
||||
</script>
|
||||
|
||||
<style></style>
|
@@ -0,0 +1,200 @@
|
||||
.maxHeight{
|
||||
height: 1170rpx;
|
||||
}
|
||||
.orderList{
|
||||
margin: 40rpx 28rpx 0;
|
||||
position: relative;
|
||||
margin-top: 0rpx;
|
||||
::v-deep .scrollView {
|
||||
height:1300rpx ;
|
||||
}
|
||||
//隐藏滚动条
|
||||
::v-deep ::-webkit-scrollbar {
|
||||
display: none;
|
||||
}
|
||||
.title{
|
||||
font-size: 30rpx;
|
||||
color:#333334 ;
|
||||
font-weight: bold;
|
||||
margin-bottom: 10rpx;
|
||||
}
|
||||
.see-more{
|
||||
font-size: 24rpx;
|
||||
text-align: center;
|
||||
margin-top: 22rpx;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
.arrow{
|
||||
margin-left: 10rpx;
|
||||
margin-top: 4rpx;
|
||||
width: 12rpx;
|
||||
height: 18rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
background-image: url('../../../static/icon15.png');
|
||||
}
|
||||
}
|
||||
.empty-data{
|
||||
height:1200rpx ;
|
||||
overflow: hidden;
|
||||
image{
|
||||
width: 400rpx;
|
||||
height: 240rpx;
|
||||
text-align: center;
|
||||
margin: 0 auto;
|
||||
display: block;
|
||||
margin-top: 27vh;
|
||||
}
|
||||
.tips{
|
||||
color: #818181;
|
||||
font-size: 28rpx;
|
||||
text-align: center;
|
||||
margin-top: 40rpx;
|
||||
margin-left: 25rpx;
|
||||
}
|
||||
}
|
||||
.notLogin{
|
||||
position: relative;
|
||||
top: 36vh;
|
||||
.toLogin-btn{
|
||||
width: 240rpx;
|
||||
height: 88rpx;
|
||||
background: #E63E32;
|
||||
border-radius: 44rpx;
|
||||
color: white;
|
||||
font-size: 30rpx;
|
||||
margin: 0 auto;
|
||||
text-align: center;
|
||||
line-height: 88rpx;
|
||||
}
|
||||
.tips{
|
||||
color:#6C6C6C ;
|
||||
font-size: 24rpx;
|
||||
text-align: center;
|
||||
margin-top: 30rpx;
|
||||
}
|
||||
}
|
||||
.order-item{
|
||||
position: relative;
|
||||
background: #FFFFFF;
|
||||
border-radius: 24rpx;
|
||||
padding:28rpx 32rpx 20rpx;
|
||||
margin-top: 20rpx;
|
||||
.orderNumber{
|
||||
font-size: 20rpx;
|
||||
color: #878787;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
.copy{
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
margin-left: 10rpx;
|
||||
}
|
||||
}
|
||||
|
||||
.area-to-area{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
margin-top:40rpx ;
|
||||
margin-left: 42rpx;
|
||||
margin-right: 42rpx;
|
||||
margin-bottom: 20rpx;
|
||||
.sendBox,.getBox{
|
||||
text-align: center;
|
||||
.sendArea,.getArea{
|
||||
font-size: 30rpx;
|
||||
font-weight: bold;
|
||||
}
|
||||
.sendName,.getName{
|
||||
font-size: 24rpx;
|
||||
color: #878787;
|
||||
}
|
||||
}
|
||||
.order-status{
|
||||
.status{
|
||||
font-size: 24rpx;
|
||||
color: #878787;
|
||||
padding-left: 20rpx;
|
||||
}
|
||||
.arrow{
|
||||
width: 160rpx;
|
||||
height: 20rpx;
|
||||
background-repeat: no-repeat;
|
||||
background-size: contain;
|
||||
}
|
||||
.green-arrow{
|
||||
background-image: url();
|
||||
}
|
||||
.red-arrow{
|
||||
background-image: url();
|
||||
}
|
||||
.gray-arrow{
|
||||
background-image: url('');
|
||||
}
|
||||
}
|
||||
}
|
||||
.order-detail{
|
||||
margin-top: 12rpx;
|
||||
overflow: hidden;
|
||||
border-bottom: 2rpx solid #F4F4F4;
|
||||
padding-bottom: 26rpx;
|
||||
.detail-item{
|
||||
overflow: hidden;
|
||||
text-overflow:ellipsis;
|
||||
white-space: nowrap;
|
||||
font-size: 24rpx;
|
||||
margin-top: 12rpx;
|
||||
uni-text,text{
|
||||
color: #878787;
|
||||
display: inline-block;
|
||||
}
|
||||
}
|
||||
}
|
||||
.btn-box{
|
||||
.btn{
|
||||
width: 144rpx;
|
||||
height: 56rpx;
|
||||
background: #FFFFFF;
|
||||
border: 2rpx solid #DDDDDD;
|
||||
border-radius: 28rpx;
|
||||
text-align: center;
|
||||
line-height: 56rpx;
|
||||
font-size: 24rpx;
|
||||
color:#303030 ;
|
||||
}
|
||||
.wait-pay{
|
||||
margin-top: 29rpx;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
|
||||
justify-content: space-between;
|
||||
.price{
|
||||
font-size: 26rpx;
|
||||
font-weight: bold;
|
||||
uni-text,text{
|
||||
color: #E84134;
|
||||
}
|
||||
}
|
||||
}
|
||||
.btn-list{
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
margin-top: 19rpx;
|
||||
.btn{
|
||||
margin-left: 20rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
.paymentStatus{
|
||||
position: absolute;
|
||||
right: 29rpx;
|
||||
top: 27rpx;
|
||||
image{
|
||||
width: 76rpx;
|
||||
height: 30rpx;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,299 @@
|
||||
<template>
|
||||
<view class="orderList" :style="{'paddingTop':capsuleBottom +'px'}">
|
||||
<view v-if="!isLogin" class="notLogin">
|
||||
<view class="toLogin-btn" @click="toLogin">登录/注册</view>
|
||||
<view class="tips">登录后可查看快递信息</view>
|
||||
</view>
|
||||
|
||||
<view v-else>
|
||||
<view v-if="!allOrderList.data.length && isLogin" class="empty-data">
|
||||
<image src="../../../static/emptyData.png"></image>
|
||||
<view class="tips">{{serchValue?'没有搜索到相关条件的运单':'暂无数据'}}</view>
|
||||
</view>
|
||||
<scroll-view v-else class="scrollView" scroll-y lower-threshold="30" @scrolltolower="LoadMoreCustomers">
|
||||
<view class="order-item" v-for="(item,index) in allOrderList.data" :key="index"
|
||||
@click='handleToOrderInfo(event,item.id,item.transportOrderId)'>
|
||||
<view class="orderNumber">
|
||||
{{[23000,22000,230011].includes(item.status)?'订':'运'}}单号:{{[23000,22000,230011].includes(item.status)?item.id:item.transportOrderId}}
|
||||
<image src="../../../static/pickUp-copy.png" class="copy"
|
||||
@click.stop="handleCopy([23000,22000,230011].includes(item.status)?item.id:item.transportOrderId)">
|
||||
</image>
|
||||
</view>
|
||||
|
||||
<view class="area-to-area">
|
||||
<view class="sendBox">
|
||||
<view class="sendArea">{{item.senderCity.name}}</view>
|
||||
<view class="sendName">{{item.senderName}}</view>
|
||||
</view>
|
||||
<view class="order-status">
|
||||
<view class="status">
|
||||
{{showOrderStatus(item.status)}}
|
||||
</view>
|
||||
<view class="arrow" :class="
|
||||
[[21000,23000,23001,23005,23008].includes(item.status)?'green-arrow':'',
|
||||
[23009,23010].includes(item.status)?'red-arrow':'',
|
||||
[230011,22000].includes(item.status)?'gray-arrow':'']
|
||||
"></view>
|
||||
</view>
|
||||
<view class="getBox">
|
||||
<view class="getArea">{{item.receiverCity.name}}</view>
|
||||
<view class="getName">{{item.receiverName}}</view>
|
||||
|
||||
</view>
|
||||
|
||||
</view>
|
||||
<view class="order-detail">
|
||||
<!-- 已取件运输信息 -->
|
||||
<view v-if="item.status ===23001 && item.transportOrderPointVOS" class="detail-item">
|
||||
<text>已取件:</text>{{ item.transportOrderPointVOS.length>0?item.transportOrderPointVOS[item.transportOrderPointVOS.length-1].info:''}}
|
||||
</view>
|
||||
<!-- 运输中信息 -->
|
||||
<view v-if="[23005,23008].includes(item.status) && item.transportOrderPointVOS"
|
||||
class="detail-item">
|
||||
<text>运送中:</text>{{item.transportOrderPointVOS.length>0?item.transportOrderPointVOS[item.transportOrderPointVOS.length-1].info:''}}
|
||||
</view>
|
||||
<!-- 已签收 -->
|
||||
<view v-if="item.status ===23009 && item.transportOrderPointVOS" class="detail-item">
|
||||
<text>已签收:</text>{{item.transportOrderPointVOS.length>0?item.transportOrderPointVOS[item.transportOrderPointVOS.length-1].info:''}}
|
||||
</view>
|
||||
<!-- 已拒收 -->
|
||||
<view v-if="item.status ===23010 && item.transportOrderPointVOS" class="detail-item">
|
||||
<text>已拒收:</text>{{item.transportOrderPointVOS.length>0?item.transportOrderPointVOS[item.transportOrderPointVOS.length-1].info:''}}
|
||||
</view>
|
||||
<!-- 预计上门时间 -->
|
||||
<view v-if="item.status ===23000 " class="detail-item">
|
||||
<text>预计上门时间:</text>{{item.estimatedStartTime}}
|
||||
</view>
|
||||
<!-- 取消时间 -->
|
||||
<view v-if="item.status ===230011 " class="detail-item"><text>取消时间:</text>{{item.updated}}
|
||||
</view>
|
||||
<!-- 预计送达时间 -->
|
||||
<view v-if="[23001,23005,23008,23010].includes(item.status) " class="detail-item">
|
||||
<text>预计送达时间:</text>{{item.estimatedArrivalTime}}
|
||||
</view>
|
||||
<!-- 已关闭时间 -->
|
||||
<view v-if="item.status ===22000 " class="detail-item"><text>关闭时间:</text>{{item.updated}}</view>
|
||||
<!-- 签收时间 -->
|
||||
<view v-if="item.status ===23009 " class="detail-item"><text>签收时间:</text>{{item.updated}}</view>
|
||||
</view>
|
||||
<!-- 功能按钮区域 -->
|
||||
<view class="btn-box">
|
||||
<view class="btn-list" @click.stop="handleSecondQi">
|
||||
<view class="btn" v-if="[23000,23001,23005,23008,23010].includes(item.status)">分享</view>
|
||||
<view class="btn" v-if='item.status===23000' @click.stop="handleOrderCancel(item.id)">取消寄件
|
||||
</view>
|
||||
<view class="btn" v-if="[22000,230011,23009].includes(item.status)"
|
||||
@click.stop="handleOrderDelete(item.id)">删除</view>
|
||||
</view>
|
||||
</view>
|
||||
<view class="paymentStatus"
|
||||
v-if="item.paymentStatus && [23001,23005,23008,23009,23010].includes(item.status)">
|
||||
<image
|
||||
:src="item.paymentStatus===1?'../../../static/daizhifu.png':'../../../static/yizhifu.png'">
|
||||
</image>
|
||||
</view>
|
||||
</view>
|
||||
<!-- 加载底部 -->
|
||||
<uni-load-more :status="status" v-if="!isShowMore && isLogin && allOrderList.data.length" />
|
||||
</scroll-view>
|
||||
</view>
|
||||
|
||||
<!-- 删除确认对话框 -->
|
||||
<uni-popup ref="popup" type="dialog" class='address-popup'>
|
||||
<uni-popup-dialog mode="base" :content="'确定是否删除此条订单?'" :animation='false' :before-close="true"
|
||||
@close="close" @confirm="confirm">
|
||||
</uni-popup-dialog>
|
||||
</uni-popup>
|
||||
|
||||
</view>
|
||||
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
reactive,
|
||||
onMounted,
|
||||
} from 'vue';
|
||||
import {
|
||||
getOrderList,
|
||||
deleteOrder
|
||||
} from '@/pages/api/order.js'
|
||||
import {
|
||||
onLoad,
|
||||
onShow,
|
||||
} from '@dcloudio/uni-app';
|
||||
import {
|
||||
handleSecondQi
|
||||
} from '@/utils/index.js'
|
||||
import {
|
||||
useStore
|
||||
} from 'vuex';
|
||||
// 获取父组件数据
|
||||
const props = defineProps({
|
||||
serchValue: {
|
||||
type: String
|
||||
}
|
||||
})
|
||||
const store = useStore(); //vuex获取、储存数据
|
||||
const users = store.state.user
|
||||
const emits = defineEmits(["@stopRefresh"]);
|
||||
let pageInfo = reactive({
|
||||
page: 1,
|
||||
pageSize: 10,
|
||||
mailType: 1
|
||||
})
|
||||
let status = ref('more') //加载状态
|
||||
let isShowMore = ref(false) //是否显示更多
|
||||
let allOrderList = reactive({
|
||||
data: []
|
||||
})
|
||||
let orderId = ref()
|
||||
let isLogin = ref('')
|
||||
let popup = ref(null)
|
||||
//胶囊底部距离头部的距离
|
||||
let capsuleBottom = ref()
|
||||
// let height = ref('570rpx')
|
||||
//关闭删除确认提示框
|
||||
const close = () => {
|
||||
popup.value.close()
|
||||
orderId.value = ''
|
||||
}
|
||||
onLoad(() => {
|
||||
uni.getSystemInfo({
|
||||
success: (res) => {
|
||||
capsuleBottom.value = uni.getMenuButtonBoundingClientRect().bottom + 52
|
||||
}
|
||||
})
|
||||
})
|
||||
onShow(() => {
|
||||
isLogin.value = uni.getStorageSync('token')
|
||||
allOrderList.data = users.indexList
|
||||
})
|
||||
onMounted(() => {
|
||||
getOrderListFunc()
|
||||
})
|
||||
//复制
|
||||
const handleCopy = (value) => {
|
||||
uni.setClipboardData({
|
||||
data: value,
|
||||
showToast: false,
|
||||
success: () => {
|
||||
uni.hideToast(); // 隐藏弹出提示
|
||||
uni.hideKeyboard(); // 隐藏软键盘
|
||||
uni.showToast({
|
||||
title: '复制成功',
|
||||
icon: 'success',
|
||||
duration: 1000
|
||||
})
|
||||
|
||||
}
|
||||
});
|
||||
}
|
||||
//确认删除订单
|
||||
const confirm = () => {
|
||||
popup.value.close()
|
||||
deleteOrder(orderId.value).then((res) => {
|
||||
pageInfo.page = 1
|
||||
pageInfo.pageSize = 10
|
||||
getOrderListFunc()
|
||||
uni.showToast({
|
||||
title: '删除成功',
|
||||
icon: 'success',
|
||||
duration: 1000
|
||||
})
|
||||
}).catch((err) => {
|
||||
uni.showToast({
|
||||
title: '网络异常',
|
||||
duration: 2000,
|
||||
icon: 'none'
|
||||
});
|
||||
})
|
||||
}
|
||||
//删除订单
|
||||
const handleOrderDelete = (id) => {
|
||||
orderId.value = id
|
||||
popup.value.open()
|
||||
}
|
||||
//跳转到取消订单页面
|
||||
const handleOrderCancel = (id) => {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/order-cancel/index?orderId=' + id
|
||||
})
|
||||
}
|
||||
const toLogin = () => {
|
||||
uni.navigateTo({
|
||||
url: '/pages/login/index'
|
||||
})
|
||||
}
|
||||
const getOrderListFunc = (flag) => {
|
||||
status.value = 'loading'
|
||||
getOrderList(pageInfo).then((res) => {
|
||||
console.log(res, '接受到了resovle')
|
||||
if (res.data) {
|
||||
console.log(res.data.items && res.data.items.length, 'res.data.items && res.data.items.length')
|
||||
status.value = (res.data.items && res.data.items.length) < 10 ? 'no-more' : 'more'
|
||||
if (flag === 'topPull') {
|
||||
allOrderList.data = allOrderList.data.concat(res.data.items ? res.data.items : [])
|
||||
} else {
|
||||
allOrderList.data = res.data.items ? res.data.items : []
|
||||
}
|
||||
}
|
||||
store.commit('user/setIndexList', allOrderList.data)
|
||||
emits('stopRefresh')
|
||||
})
|
||||
}
|
||||
|
||||
//根据状态去显示对应的运单状态文案
|
||||
const showOrderStatus = (status) => {
|
||||
switch (status) {
|
||||
case 21000:
|
||||
return '待支付';
|
||||
case 23000:
|
||||
return '待取件';
|
||||
case 230011:
|
||||
return '已取消';
|
||||
case 23001:
|
||||
return '已取件';
|
||||
case 23005:
|
||||
return '运送中';
|
||||
case 22000:
|
||||
return '已关闭';
|
||||
case 23008:
|
||||
return '派送中';
|
||||
case 23009:
|
||||
return '已签收';
|
||||
case 23010:
|
||||
return '已拒收';
|
||||
}
|
||||
}
|
||||
//跳转到订单详情页面
|
||||
const handleToOrderInfo = (event, id, transportOrderId) => {
|
||||
uni.navigateTo({
|
||||
url: '/subPages/order-info/index?orderId=' + id + '&transportOrderId=' + transportOrderId
|
||||
});
|
||||
store.commit('user/setIsToOrderInfo', true)
|
||||
}
|
||||
|
||||
//下拉加载更多
|
||||
const LoadMoreCustomers = () => {
|
||||
if (status.value === 'no-more') {
|
||||
return
|
||||
}
|
||||
pageInfo.page = pageInfo.page + 1
|
||||
getOrderListFunc('topPull')
|
||||
}
|
||||
//
|
||||
const indexGetOrderListFunc = (params) => {
|
||||
pageInfo.page = 1
|
||||
pageInfo = Object.assign({}, pageInfo, params)
|
||||
getOrderListFunc()
|
||||
}
|
||||
// 暴漏给父组件
|
||||
defineExpose({
|
||||
indexGetOrderListFunc,
|
||||
LoadMoreCustomers
|
||||
});
|
||||
</script>
|
||||
|
||||
<style src="./orderList.scss" lang="scss" scoped></style>
|
18
project-wl-yonghuduan-uniapp-vue3/pages/pickup/index.scss
Normal file
18
project-wl-yonghuduan-uniapp-vue3/pages/pickup/index.scss
Normal file
@@ -0,0 +1,18 @@
|
||||
|
||||
.tabScroll {
|
||||
.scroll-row-item {
|
||||
margin-right: 80rpx;
|
||||
}
|
||||
}
|
||||
.boxTop{
|
||||
position: fixed;
|
||||
top: 0;
|
||||
width: 100%;
|
||||
z-index: 2;
|
||||
box-shadow: 0 4rpx 12rpx 0 rgba(0,0,0,0.03);
|
||||
}
|
||||
.pickup{
|
||||
min-height: 100vh!important;
|
||||
background-color: #F3F5F9;
|
||||
}
|
||||
|
108
project-wl-yonghuduan-uniapp-vue3/pages/pickup/index.vue
Normal file
108
project-wl-yonghuduan-uniapp-vue3/pages/pickup/index.vue
Normal file
@@ -0,0 +1,108 @@
|
||||
<!-- 取件页面 -->
|
||||
<template>
|
||||
<view class="pickup">
|
||||
<!-- 搜索nav -->
|
||||
<SearchPage @handleSearch="handleSearch"></SearchPage>
|
||||
<!-- end -->
|
||||
<view class="boxTop" :style="{'paddingTop':capsuleBottom +'px'}">
|
||||
<!-- tab切换 -->
|
||||
<UniTab :tabBars="tabBars" ref="tab" @getTabIndex="getTabIndex" :staticNum="staticNum.data"></UniTab>
|
||||
</view>
|
||||
<!-- 运单列表 -->
|
||||
<OrderList ref='orderListRef' :serchValue="serchValue" @stopRefresh="stopRefreshFunc"></OrderList>
|
||||
</view>
|
||||
<!-- end -->
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import {
|
||||
ref,
|
||||
reactive,
|
||||
} from 'vue';
|
||||
import {
|
||||
onLoad,
|
||||
onShow
|
||||
} from '@dcloudio/uni-app';
|
||||
|
||||
// 基本数据
|
||||
import {
|
||||
DeliveryData
|
||||
} from '@/utils/commonData.js';
|
||||
import {
|
||||
getGoodsNum
|
||||
} from '@/pages/api/order.js'
|
||||
// 导入组件
|
||||
// 搜索组件
|
||||
import SearchPage from '@/components/uni-search/index.vue';
|
||||
// tab切换
|
||||
import UniTab from '@/components/uni-tab/index.vue';
|
||||
//运单列表
|
||||
import OrderList from './components/orderList';
|
||||
// 筛选
|
||||
|
||||
const emit = defineEmits(''); //子组件向父组件事件传递
|
||||
const tab = ref();
|
||||
const tabBars = DeliveryData;
|
||||
let tabIndex = ref(0); //当前tab
|
||||
let staticNum = reactive({
|
||||
data: [0, 0]
|
||||
})
|
||||
// 存储已选内容, 因为这个列表是增删很频繁的,所以选用map而不是数组,key对应的是数据的下标。至于value存放什么,完全由自己定
|
||||
let selected = reactive(new Map());
|
||||
let orderListRef = ref() //列表组件引用
|
||||
//设备栏高度
|
||||
let deviceNavHeight = ref()
|
||||
//胶囊顶部距离头部的距离
|
||||
let capsuleTop = ref()
|
||||
//胶囊底部距离头部的距离
|
||||
let capsuleBottom = ref()
|
||||
//导航栏高度
|
||||
let all = ref()
|
||||
//胶囊高度
|
||||
let capsuleHeight = ref()
|
||||
let serchValue = ref()//头部搜索栏的关键字
|
||||
// ------生命周期------
|
||||
onShow(() => {
|
||||
getStaticNum()
|
||||
orderListRef.value && getTabIndex(tabIndex.value)
|
||||
});
|
||||
onLoad(() => {
|
||||
uni.getSystemInfo({
|
||||
success: (res) => {
|
||||
deviceNavHeight.value = res.statusBarHeight
|
||||
capsuleTop.value = uni.getMenuButtonBoundingClientRect().top
|
||||
capsuleBottom.value = uni.getMenuButtonBoundingClientRect().bottom
|
||||
all.value = (capsuleTop.value + capsuleBottom.value - deviceNavHeight.value) + 'px'
|
||||
capsuleHeight.value = uni.getMenuButtonBoundingClientRect().height
|
||||
|
||||
}
|
||||
})
|
||||
})
|
||||
// ------定义方法------
|
||||
|
||||
// 搜索
|
||||
const handleSearch = (index) => {
|
||||
serchValue.value = index.value
|
||||
orderListRef.value.indexGetOrderListFunc({
|
||||
keyword: index.value
|
||||
}) //触发子组件获取列表数据的方法
|
||||
};
|
||||
// 获取寄件和收件的数量
|
||||
const getStaticNum = () => {
|
||||
getGoodsNum().then((res) => {
|
||||
if(res.code === 200){
|
||||
staticNum.data = [res.data['1'],res.data['2']]
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
// 获取tab切换当前的index
|
||||
const getTabIndex = index => {
|
||||
tabIndex.value = index
|
||||
orderListRef.value.indexGetOrderListFunc({
|
||||
mailType: index + 1,
|
||||
}) //触发子组件获取列表数据的方法
|
||||
};
|
||||
</script>
|
||||
<style src="../../styles/expressage.scss" lang="scss" scoped></style>
|
||||
<style src="./index.scss" lang="scss" scoped></style>
|
Reference in New Issue
Block a user