完善下单逻辑及其ui展示,修复支付倒计时显示错误bug
This commit is contained in:
289
pages/address/address-map-select.uvue
Normal file
289
pages/address/address-map-select.uvue
Normal file
@@ -0,0 +1,289 @@
|
||||
<template>
|
||||
<view class="map-page">
|
||||
<view class="search-card">
|
||||
<input v-model="keyword" class="search-input" placeholder="搜索小区、医院、养老院、街道" @input="handleKeywordInput" />
|
||||
</view>
|
||||
|
||||
<map
|
||||
class="address-map"
|
||||
:latitude="latitude"
|
||||
:longitude="longitude"
|
||||
:markers="markers"
|
||||
:show-location="true"
|
||||
:scale="16"
|
||||
></map>
|
||||
|
||||
<scroll-view class="poi-scroll" scroll-y="true">
|
||||
<view class="poi-card">
|
||||
<text class="poi-title">附近地址</text>
|
||||
<view
|
||||
v-for="item in poiList"
|
||||
:key="item.id"
|
||||
:class="['poi-item', selectedPoiId == item.id ? 'poi-item-selected' : '']"
|
||||
@click="selectPoi(item.id)"
|
||||
>
|
||||
<text class="poi-name">{{ item.name }}</text>
|
||||
<text class="poi-address">{{ item.address }}</text>
|
||||
</view>
|
||||
</view>
|
||||
</scroll-view>
|
||||
|
||||
<view class="bottom-bar">
|
||||
<button class="confirm-btn" @click="confirmAddress">确认地址</button>
|
||||
</view>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup lang="uts">
|
||||
import { ref } from 'vue'
|
||||
import { onLoad } from '@dcloudio/uni-app'
|
||||
import { HomeServiceSelectedAddressType } from '@/types/home-service.uts'
|
||||
|
||||
const SELECTED_KEY = 'hss_selected_service_address'
|
||||
const MAP_DRAFT_KEY = 'hss_service_address_map_draft'
|
||||
|
||||
type MapPoiItemType = {
|
||||
id: string
|
||||
name: string
|
||||
address: string
|
||||
latitude: number
|
||||
longitude: number
|
||||
}
|
||||
|
||||
const keyword = ref('')
|
||||
const latitude = ref(24.28859)
|
||||
const longitude = ref(116.12264)
|
||||
const selectedPoiId = ref('poi-current')
|
||||
const poiList = ref<Array<MapPoiItemType>>([])
|
||||
const markers = ref<Array<UTSJSONObject>>([])
|
||||
|
||||
function buildMarkers(): Array<UTSJSONObject> {
|
||||
const result: Array<UTSJSONObject> = []
|
||||
const selected = getSelectedPoi()
|
||||
if (selected == null) {
|
||||
return result
|
||||
}
|
||||
const marker = new UTSJSONObject()
|
||||
marker.set('id', 1)
|
||||
marker.set('latitude', selected.latitude)
|
||||
marker.set('longitude', selected.longitude)
|
||||
marker.set('title', selected.name)
|
||||
result.push(marker)
|
||||
return result
|
||||
}
|
||||
|
||||
function createDefaultPoiList(selected: HomeServiceSelectedAddressType | null): Array<MapPoiItemType> {
|
||||
const locationName = selected != null && selected.locationName != null && selected.locationName != '' ? selected.locationName : '当前定位'
|
||||
const locationAddress = selected != null && selected.locationAddress != null && selected.locationAddress != '' ? selected.locationAddress : '请通过系统地图继续完善地址搜索'
|
||||
const baseLatitude = selected != null && selected.latitude != 0 ? selected.latitude : latitude.value
|
||||
const baseLongitude = selected != null && selected.longitude != 0 ? selected.longitude : longitude.value
|
||||
const result: Array<MapPoiItemType> = []
|
||||
result.push({
|
||||
id: 'poi-current',
|
||||
name: locationName,
|
||||
address: locationAddress,
|
||||
latitude: baseLatitude,
|
||||
longitude: baseLongitude
|
||||
})
|
||||
result.push({
|
||||
id: 'poi-near-1',
|
||||
name: locationName + '附近入口',
|
||||
address: locationAddress,
|
||||
latitude: baseLatitude + 0.0006,
|
||||
longitude: baseLongitude + 0.0006
|
||||
})
|
||||
result.push({
|
||||
id: 'poi-near-2',
|
||||
name: locationName + '周边推荐',
|
||||
address: locationAddress,
|
||||
latitude: baseLatitude - 0.0006,
|
||||
longitude: baseLongitude - 0.0006
|
||||
})
|
||||
return result
|
||||
}
|
||||
|
||||
function getSelectedPoi(): MapPoiItemType | null {
|
||||
for (let i = 0; i < poiList.value.length; i++) {
|
||||
if (poiList.value[i].id == selectedPoiId.value) {
|
||||
return poiList.value[i]
|
||||
}
|
||||
}
|
||||
return poiList.value.length > 0 ? poiList.value[0] : null
|
||||
}
|
||||
|
||||
function selectPoi(poiId: string): void {
|
||||
selectedPoiId.value = poiId
|
||||
const selected = getSelectedPoi()
|
||||
if (selected == null) {
|
||||
return
|
||||
}
|
||||
latitude.value = selected.latitude
|
||||
longitude.value = selected.longitude
|
||||
markers.value = buildMarkers()
|
||||
}
|
||||
|
||||
function handleKeywordInput(): void {
|
||||
if (keyword.value.trim() == '') {
|
||||
return
|
||||
}
|
||||
const base = getSelectedPoi()
|
||||
if (base == null) {
|
||||
return
|
||||
}
|
||||
const dynamicList: Array<MapPoiItemType> = []
|
||||
dynamicList.push({
|
||||
id: 'poi-search-1',
|
||||
name: keyword.value.trim(),
|
||||
address: base.address,
|
||||
latitude: base.latitude,
|
||||
longitude: base.longitude
|
||||
})
|
||||
dynamicList.push(base)
|
||||
poiList.value = dynamicList
|
||||
selectedPoiId.value = 'poi-search-1'
|
||||
markers.value = buildMarkers()
|
||||
}
|
||||
|
||||
function confirmAddress(): void {
|
||||
const selected = getSelectedPoi()
|
||||
if (selected == null) {
|
||||
uni.showToast({ title: '请先选择地址', icon: 'none' })
|
||||
return
|
||||
}
|
||||
const draft = {
|
||||
addressId: '',
|
||||
userId: '',
|
||||
isDefault: true,
|
||||
contactName: '',
|
||||
contactPhone: '',
|
||||
phone: '',
|
||||
addressName: selected.name,
|
||||
locationName: selected.name,
|
||||
addressDetail: selected.address,
|
||||
locationAddress: selected.address,
|
||||
houseNumber: '',
|
||||
doorNo: '',
|
||||
fullAddress: selected.address,
|
||||
latitude: selected.latitude,
|
||||
longitude: selected.longitude,
|
||||
remark: '',
|
||||
coordinateType: 'gcj02',
|
||||
createdAt: Date.now(),
|
||||
updatedAt: Date.now()
|
||||
} as HomeServiceSelectedAddressType
|
||||
uni.setStorageSync(MAP_DRAFT_KEY, JSON.stringify(draft))
|
||||
uni.navigateBack()
|
||||
}
|
||||
|
||||
onLoad(() => {
|
||||
const selected = uni.getStorageSync(SELECTED_KEY) as HomeServiceSelectedAddressType | null
|
||||
if (selected != null && selected.latitude != 0) {
|
||||
latitude.value = selected.latitude
|
||||
longitude.value = selected.longitude
|
||||
}
|
||||
poiList.value = createDefaultPoiList(selected)
|
||||
markers.value = buildMarkers()
|
||||
})
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.map-page {
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
background: #f4f6f8;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 24rpx 24rpx 180rpx;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.search-card,
|
||||
.poi-card {
|
||||
background: #ffffff;
|
||||
border-radius: 28rpx;
|
||||
padding: 24rpx;
|
||||
box-shadow: 0 12rpx 28rpx rgba(15, 23, 42, 0.05);
|
||||
}
|
||||
|
||||
.search-input {
|
||||
background: #f8fafc;
|
||||
border-radius: 999rpx;
|
||||
padding: 18rpx 24rpx;
|
||||
font-size: 26rpx;
|
||||
color: #1f2937;
|
||||
}
|
||||
|
||||
.address-map {
|
||||
width: 100%;
|
||||
height: 45vh;
|
||||
border-radius: 28rpx;
|
||||
overflow: hidden;
|
||||
margin-top: 20rpx;
|
||||
margin-bottom: 20rpx;
|
||||
}
|
||||
|
||||
.poi-scroll {
|
||||
flex: 1;
|
||||
min-height: 0;
|
||||
}
|
||||
|
||||
.poi-title,
|
||||
.poi-name {
|
||||
color: #1f2937;
|
||||
}
|
||||
|
||||
.poi-title {
|
||||
font-size: 28rpx;
|
||||
font-weight: 600;
|
||||
margin-bottom: 12rpx;
|
||||
}
|
||||
|
||||
.poi-item {
|
||||
padding: 18rpx 0;
|
||||
border-top: 1rpx solid #eef2f7;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 8rpx;
|
||||
}
|
||||
|
||||
.poi-item-selected {
|
||||
border-left: 6rpx solid #f97316;
|
||||
padding-left: 18rpx;
|
||||
}
|
||||
|
||||
.poi-name {
|
||||
font-size: 26rpx;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.poi-address {
|
||||
font-size: 24rpx;
|
||||
color: #4b5563;
|
||||
line-height: 1.6;
|
||||
}
|
||||
|
||||
.bottom-bar {
|
||||
position: fixed;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
padding: 20rpx 24rpx 36rpx;
|
||||
background: rgba(244, 246, 248, 0.96);
|
||||
box-shadow: 0 -8rpx 24rpx rgba(15, 23, 42, 0.05);
|
||||
}
|
||||
|
||||
.confirm-btn {
|
||||
width: 100%;
|
||||
height: 88rpx;
|
||||
line-height: 88rpx;
|
||||
border-radius: 999rpx;
|
||||
background: linear-gradient(135deg, #ff8a65 0%, #ff7043 100%);
|
||||
color: #ffffff;
|
||||
font-size: 30rpx;
|
||||
font-weight: 600;
|
||||
border: none;
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user