319 lines
7.3 KiB
Plaintext
319 lines
7.3 KiB
Plaintext
<template>
|
|
<view class="address-edit-page">
|
|
<view class="form-group">
|
|
<view class="form-item">
|
|
<text class="label">收货人</text>
|
|
<input class="input" v-model="formData.name" placeholder="请填写收货人姓名" />
|
|
</view>
|
|
<view class="form-item">
|
|
<text class="label">手机号码</text>
|
|
<input class="input" v-model="formData.phone" type="number" maxlength="11" placeholder="请填写手机号码" />
|
|
</view>
|
|
<view class="form-item">
|
|
<text class="label">所在地区</text>
|
|
<input class="input" v-model="regionString" placeholder="省市区县、乡镇等" />
|
|
</view>
|
|
<view class="form-item">
|
|
<text class="label">详细地址</text>
|
|
<input class="input" v-model="formData.detail" placeholder="街道、楼牌号等" />
|
|
</view>
|
|
</view>
|
|
|
|
<view class="form-group">
|
|
<view class="form-item">
|
|
<text class="label">标签</text>
|
|
<view class="tags-container">
|
|
<text
|
|
v-for="tag in tags"
|
|
:key="tag"
|
|
class="tag-item"
|
|
:class="{ active: formData.label === tag }"
|
|
@click="selectTag(tag)"
|
|
>{{ tag }}</text>
|
|
</view>
|
|
</view>
|
|
<view class="form-item switch-item">
|
|
<text class="label">设为默认收货地址</text>
|
|
<switch :checked="formData.isDefault" color="#ff5000" @change="onSwitchChange" />
|
|
</view>
|
|
</view>
|
|
|
|
<view class="footer-btn">
|
|
<button class="save-btn" @click="saveAddress">保存</button>
|
|
<button v-if="isEdit" class="delete-btn" @click="deleteAddress">删除收货地址</button>
|
|
</view>
|
|
</view>
|
|
</template>
|
|
|
|
<script setup lang="uts">
|
|
import { ref, reactive, computed } from 'vue'
|
|
import { onLoad } from '@dcloudio/uni-app'
|
|
|
|
type Address = {
|
|
id: string
|
|
name: string
|
|
phone: string
|
|
province: string
|
|
city: string
|
|
district: string
|
|
detail: string
|
|
isDefault: boolean
|
|
label?: string
|
|
}
|
|
|
|
const isEdit = ref(false)
|
|
const addressId = ref('')
|
|
const regionString = ref('')
|
|
const tags = ['家', '公司', '学校']
|
|
|
|
const formData = reactive({
|
|
name: '',
|
|
phone: '',
|
|
detail: '',
|
|
isDefault: false,
|
|
label: ''
|
|
} as {
|
|
name: string
|
|
phone: string
|
|
detail: string
|
|
isDefault: boolean
|
|
label: string
|
|
})
|
|
|
|
onLoad((options) => {
|
|
if (options['id']) {
|
|
isEdit.value = true
|
|
addressId.value = options['id'] as string
|
|
loadAddress(addressId.value)
|
|
}
|
|
})
|
|
|
|
const loadAddress = (id: string) => {
|
|
const storedAddresses = uni.getStorageSync('addresses')
|
|
if (storedAddresses) {
|
|
const addresses = JSON.parse(storedAddresses as string) as Address[]
|
|
const address = addresses.find(item => item.id === id)
|
|
if (address) {
|
|
formData.name = address.name
|
|
formData.phone = address.phone
|
|
formData.detail = address.detail
|
|
formData.isDefault = address.isDefault
|
|
formData.label = address.label || ''
|
|
regionString.value = `${address.province} ${address.city} ${address.district}`.trim()
|
|
}
|
|
}
|
|
}
|
|
|
|
const selectTag = (tag: string) => {
|
|
if (formData.label === tag) {
|
|
formData.label = ''
|
|
} else {
|
|
formData.label = tag
|
|
}
|
|
}
|
|
|
|
const onSwitchChange = (e: UniSwitchChangeEvent) => {
|
|
formData.isDefault = e.detail.value
|
|
}
|
|
|
|
const saveAddress = () => {
|
|
if (!formData.name) {
|
|
uni.showToast({ title: '请填写收货人', icon: 'none' })
|
|
return
|
|
}
|
|
if (!formData.phone) {
|
|
uni.showToast({ title: '请填写手机号码', icon: 'none' })
|
|
return
|
|
}
|
|
if (!regionString.value) {
|
|
uni.showToast({ title: '请填写所在地区', icon: 'none' })
|
|
return
|
|
}
|
|
if (!formData.detail) {
|
|
uni.showToast({ title: '请填写详细地址', icon: 'none' })
|
|
return
|
|
}
|
|
|
|
// 简单解析地区(这里简化处理,实际应使用选择器)
|
|
const regions = regionString.value.split(' ')
|
|
const province = regions[0] || ''
|
|
const city = regions[1] || ''
|
|
const district = regions.slice(2).join(' ') || ''
|
|
|
|
const storedAddresses = uni.getStorageSync('addresses')
|
|
let addresses: Address[] = []
|
|
if (storedAddresses) {
|
|
try {
|
|
addresses = JSON.parse(storedAddresses as string) as Address[]
|
|
} catch (e) {
|
|
addresses = []
|
|
}
|
|
}
|
|
|
|
// 如果设为默认,取消其他默认
|
|
if (formData.isDefault) {
|
|
addresses.forEach(item => {
|
|
item.isDefault = false
|
|
})
|
|
}
|
|
|
|
if (isEdit.value) {
|
|
const index = addresses.findIndex(item => item.id === addressId.value)
|
|
if (index !== -1) {
|
|
addresses[index] = {
|
|
...addresses[index],
|
|
name: formData.name,
|
|
phone: formData.phone,
|
|
province: province,
|
|
city: city,
|
|
district: district,
|
|
detail: formData.detail,
|
|
isDefault: formData.isDefault,
|
|
label: formData.label
|
|
}
|
|
}
|
|
} else {
|
|
const newAddress: Address = {
|
|
id: `addr_${Date.now()}`,
|
|
name: formData.name,
|
|
phone: formData.phone,
|
|
province: province,
|
|
city: city,
|
|
district: district,
|
|
detail: formData.detail,
|
|
isDefault: formData.isDefault,
|
|
label: formData.label
|
|
}
|
|
// 如果是第一个地址,自动设为默认
|
|
if (addresses.length === 0) {
|
|
newAddress.isDefault = true
|
|
}
|
|
addresses.push(newAddress)
|
|
}
|
|
|
|
uni.setStorageSync('addresses', JSON.stringify(addresses))
|
|
|
|
uni.showToast({
|
|
title: '保存成功',
|
|
icon: 'success'
|
|
})
|
|
|
|
setTimeout(() => {
|
|
uni.navigateBack()
|
|
}, 1500)
|
|
}
|
|
|
|
const deleteAddress = () => {
|
|
uni.showModal({
|
|
title: '提示',
|
|
content: '确定要删除该地址吗?',
|
|
success: (res) => {
|
|
if (res.confirm) {
|
|
const storedAddresses = uni.getStorageSync('addresses')
|
|
if (storedAddresses) {
|
|
let addresses = JSON.parse(storedAddresses as string) as Address[]
|
|
addresses = addresses.filter(item => item.id !== addressId.value)
|
|
uni.setStorageSync('addresses', JSON.stringify(addresses))
|
|
|
|
uni.showToast({
|
|
title: '删除成功',
|
|
icon: 'success'
|
|
})
|
|
|
|
setTimeout(() => {
|
|
uni.navigateBack()
|
|
}, 1500)
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
</script>
|
|
|
|
<style>
|
|
.address-edit-page {
|
|
min-height: 100vh;
|
|
background-color: #f5f5f5;
|
|
padding-top: 15px;
|
|
}
|
|
|
|
.form-group {
|
|
background-color: white;
|
|
margin-bottom: 15px;
|
|
padding: 0 15px;
|
|
}
|
|
|
|
.form-item {
|
|
display: flex;
|
|
align-items: center;
|
|
border-bottom: 1px solid #f5f5f5;
|
|
padding: 15px 0;
|
|
}
|
|
|
|
.form-item:last-child {
|
|
border-bottom: none;
|
|
}
|
|
|
|
.label {
|
|
width: 80px;
|
|
font-size: 15px;
|
|
color: #333;
|
|
}
|
|
|
|
.input {
|
|
flex: 1;
|
|
font-size: 15px;
|
|
color: #333;
|
|
}
|
|
|
|
.switch-item {
|
|
justify-content: space-between;
|
|
}
|
|
|
|
.tags-container {
|
|
flex: 1;
|
|
display: flex;
|
|
flex-wrap: wrap;
|
|
}
|
|
|
|
.tag-item {
|
|
font-size: 12px;
|
|
color: #666;
|
|
border: 1px solid #ddd;
|
|
padding: 4px 12px;
|
|
border-radius: 15px;
|
|
margin-right: 10px;
|
|
}
|
|
|
|
.tag-item.active {
|
|
background-color: #ff5000;
|
|
color: white;
|
|
border-color: #ff5000;
|
|
}
|
|
|
|
.footer-btn {
|
|
margin-top: 30px;
|
|
padding: 0 15px;
|
|
}
|
|
|
|
.save-btn {
|
|
background-color: #ff5000;
|
|
color: white;
|
|
border-radius: 25px;
|
|
font-size: 16px;
|
|
height: 44px;
|
|
line-height: 44px;
|
|
border: none;
|
|
margin-bottom: 15px;
|
|
}
|
|
|
|
.delete-btn {
|
|
background-color: white;
|
|
color: #333;
|
|
border-radius: 25px;
|
|
font-size: 16px;
|
|
height: 44px;
|
|
line-height: 44px;
|
|
border: 1px solid #ddd;
|
|
}
|
|
</style> |