HomeLease/pages/lease/lease.vue
2025-08-19 09:36:20 +08:00

793 lines
18 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<template>
<view class="lease-page">
<!-- 头部区域 -->
<custom-nav-bar2 color="#FFDECB" title="租赁申请"></custom-nav-bar2>
<view class="header">
<image
:src="commonEnum.FIRE_BACKGROUND_FULL"
class="fire-background-full"
mode="aspectFill"
></image>
</view>
<!-- 主要内容区域 -->
<view class="main-content">
<!-- 租赁信息表单 -->
<view class="form-section">
<view class="section-header">
<view class="section-indicator"></view>
<text class="section-title">填写租赁信息</text>
</view>
<view class="form-fields">
<!-- 姓名 -->
<view class="form-item">
<text class="field-label">姓名</text>
<input v-model="formData.name" class="field-input" placeholder="请输入姓名" />
</view>
<!-- 手机号 -->
<view class="form-item">
<text class="field-label">手机号</text>
<input
v-model="formData.phone"
class="field-input"
placeholder="请填写手机号"
type="number"
/>
</view>
<!-- 地址 -->
<view class="form-item address-form-item">
<text class="field-label">地址</text>
<view class="address-input-wrapper">
<input
v-model="formData.address"
class="address-input"
placeholder="请输入或选择收货地址"
@focus="onAddressFocus"
/>
<view
class="map-icon-wrapper"
title="点击获取定位并打开地图"
@click="openMapWithLocation"
>
<text class="map-icon">🗺️</text>
</view>
</view>
</view>
<!-- 当前定位 -->
<view class="location-suggestion">
<view v-if="currentLocation" class="location-card">
<view class="location-content">
<view class="location-header">
<text class="location-title">当前定位</text>
</view>
<text class="location-address">{{ currentLocation.address }}</text>
</view>
<button class="use-location-btn" @click="useCurrentLocation">使用</button>
</view>
<view v-else class="location-card">
<view class="location-content">
<view class="location-header">
<text class="location-title">获取位置</text>
<text class="location-company">点击获取当前位置</text>
</view>
<text class="location-address">需要定位权限</text>
</view>
<button class="get-location-btn" @click="getCurrentLocation">获取</button>
</view>
</view>
<!-- 详细位置 -->
<view class="form-item">
<text class="field-label">详细位置</text>
<input v-model="formData.detailAddress" class="field-input" placeholder="例:6栋201室" />
</view>
<!-- 租赁设备 -->
<view class="form-item">
<text class="field-label">租赁设备</text>
<view class="selector" @click="selectEquipment">
<text class="selector-text">{{ formData.equipment || '选择设备类型' }}</text>
<text class="arrow-icon">></text>
</view>
</view>
<!-- 租赁周期 -->
<view class="form-item">
<text class="field-label">租赁周期</text>
<view class="selector" @click="selectPeriod">
<text class="selector-text">{{ formData.period || '1年' }}</text>
<text class="arrow-icon">></text>
</view>
</view>
</view>
</view>
<!-- 支付区域 -->
<view class="payment-section">
<button class="pay-button" @click="handlePayment">立即支付 ¥{{ totalAmount }}</button>
<view class="payment-details">
<view class="details-header" @click="toggleDetails">
<text class="details-title">明细</text>
<view :class="{ expanded: showDetails }" class="arrow-wrapper">
<image :src="commonEnum.DOWN_ARROW" class="details-arrow"></image>
</view>
</view>
<view v-if="showDetails" class="details-content">
<view class="detail-item">
<text class="detail-label">租金</text>
<text class="detail-value">¥{{ totalAmount }}</text>
</view>
</view>
</view>
</view>
</view>
<!-- 底部导航已由系统tabBar处理 -->
</view>
</template>
<script>
import commonEnum from '../../enum/commonEnum'
import { getLocationWithPermission, handleLocationError } from '@/utils/permission.js'
export default {
name: 'LeasePage',
computed: {
commonEnum() {
return commonEnum
},
},
onLoad() {
// 页面加载时获取当前位置
//this.getCurrentLocation()
},
data() {
return {
formData: {
name: '张珊珊',
phone: '',
address: '',
detailAddress: '',
equipment: '',
period: '1年',
},
currentLocation: null,
showDetails: false,
totalAmount: '100.10',
isMapTriggered: false,
}
},
methods: {
// 获取当前位置信息
async getCurrentLocation() {
try {
uni.showLoading({
title: '获取位置中...',
})
const location = await getLocationWithPermission()
console.log('位置信息:', location)
// 直接使用坐标信息,简化处理
this.currentLocation = {
address: `纬度: ${location.latitude.toFixed(6)}, 经度: ${location.longitude.toFixed(6)}`,
latitude: location.latitude,
longitude: location.longitude,
}
uni.hideLoading()
// 如果不是通过地图图标触发的,才显示提示
if (!this.isMapTriggered) {
uni.showToast({
title: '位置获取成功,点击地图图标查看',
icon: 'success',
duration: 3000,
})
}
return this.currentLocation
} catch (err) {
uni.hideLoading()
handleLocationError(err)
throw err
}
},
// 地址输入框获得焦点
onAddressFocus() {
// 可以在这里添加地址输入的相关逻辑
console.log('地址输入框获得焦点')
},
// 点击地图图标,自动获取定位并打开地图
async openMapWithLocation() {
try {
this.isMapTriggered = true
// 先获取当前位置
await this.getCurrentLocation()
// 获取成功后自动打开地图
this.openMap()
} catch (err) {
console.error('获取位置失败:', err)
uni.showToast({
title: '获取位置失败',
icon: 'error',
})
} finally {
this.isMapTriggered = false
}
},
selectAddress() {
// 如果有当前定位信息,打开地图
if (this.currentLocation && this.currentLocation.latitude && this.currentLocation.longitude) {
this.openMap()
} else {
// 没有定位信息,先获取位置
uni.showModal({
title: '提示',
content: '需要先获取当前位置才能打开地图,是否现在获取?',
success: res => {
if (res.confirm) {
this.getCurrentLocation()
}
},
})
}
},
// 打开地图
openMap() {
const { latitude, longitude } = this.currentLocation
// 显示加载提示
uni.showLoading({
title: '打开地图中...',
})
// 尝试打开系统地图应用
// #ifdef APP-PLUS
uni.openLocation({
latitude: latitude,
longitude: longitude,
name: '当前位置',
address: this.currentLocation.address || '未知地址',
scale: 18,
success: () => {
uni.hideLoading()
console.log('地图打开成功')
},
fail: err => {
uni.hideLoading()
console.error('打开地图失败:', err)
this.showMapInApp()
},
})
// #endif
// #ifdef MP-WEIXIN
uni.openLocation({
latitude: latitude,
longitude: longitude,
name: '当前位置',
address: this.currentLocation.address || '未知地址',
scale: 18,
success: () => {
uni.hideLoading()
},
fail: err => {
uni.hideLoading()
console.error('打开地图失败:', err)
this.showMapInApp()
},
})
// #endif
// #ifdef H5
// H5环境下使用在线地图
uni.hideLoading()
const mapUrl = `https://uri.amap.com/marker?position=${longitude},${latitude}&name=${encodeURIComponent('当前位置')}&address=${encodeURIComponent(this.currentLocation.address || '未知地址')}`
window.open(mapUrl, '_blank')
// #endif
},
// 在应用内显示地图(备用方案)
showMapInApp() {
uni.showModal({
title: '地图功能',
content: `${this.currentLocation.address}\n\n经纬度${this.currentLocation.latitude.toFixed(6)}, ${this.currentLocation.longitude.toFixed(6)}`,
showCancel: false,
confirmText: '确定',
})
},
useCurrentLocation() {
if (!this.currentLocation) {
this.getCurrentLocation()
return
}
this.formData.address = this.currentLocation.address
uni.showToast({
title: '已使用当前定位',
icon: 'success',
})
},
selectEquipment() {
// 选择设备逻辑
uni.showActionSheet({
itemList: ['节能灶', '燃烧器', '燃气灶', '电磁炉'],
success: res => {
const equipmentList = ['节能灶', '燃烧器', '燃气灶', '电磁炉']
this.formData.equipment = equipmentList[res.tapIndex]
},
})
},
selectPeriod() {
// 选择租赁周期逻辑
uni.showActionSheet({
itemList: ['1个月', '3个月', '6个月', '1年', '2年'],
success: res => {
const periodList = ['1个月', '3个月', '6个月', '1年', '2年']
this.formData.period = periodList[res.tapIndex]
},
})
},
toggleDetails() {
this.showDetails = !this.showDetails
},
handlePayment() {
// 支付逻辑
uni.showModal({
title: '确认支付',
content: `确认支付 ¥${this.totalAmount} 吗?`,
success: res => {
if (res.confirm) {
uni.showToast({
title: '支付成功',
icon: 'success',
})
}
},
})
},
// 页面跳转现在由系统tabBar处理
goToHome() {
uni.switchTab({
url: '/pages/index/index',
})
},
goToProfile() {
uni.switchTab({
url: '/pages/profile/profile',
})
},
},
}
</script>
<style lang="scss" scoped>
.lease-page {
position: relative;
background: #f3f5f6;
border: #120d0d solid 2rpx;
}
// 头部区域
.header {
position: relative;
//border: #120d0d solid 2rpx;
background: #fedfcd;
.fire-background-full {
position: relative;
left: 30rpx;
width: 750rpx;
height: 592rpx;
}
}
// 主要内容区域
.main-content {
position: relative;
top: -220rpx;
background: #ffffff;
border-radius: 40rpx 40rpx 0 0;
margin: 0 30rpx;
padding: 40rpx;
min-height: 60vh;
//border: #120d0d solid 2rpx;
//box-shadow: 0 10rpx 30rpx rgba(0, 0, 0, 0.3);
}
// 表单区域
.form-section {
margin-bottom: 60rpx;
//border: #0c1387 solid 2rpx;
.section-header {
display: flex;
align-items: center;
margin-bottom: 40rpx;
//border: red solid 2rpx;
.section-indicator {
width: 8rpx;
height: 40rpx;
background: #f15a04;
border-radius: 4rpx;
margin-right: 20rpx;
}
.section-title {
font-size: 32rpx;
font-weight: bold;
color: #333;
}
}
}
.form-fields {
.form-item {
//margin-bottom: 40rpx;
display: flex;
align-items: center;
border-bottom: 1rpx solid #d8d8d8;
.field-label {
display: block;
font-size: 28rpx;
color: #333;
font-weight: 400;
flex: 1;
//border: 2rpx solid #d81313;
}
.field-input {
//border: 2rpx solid #d81313;
flex: 3;
height: 80rpx;
padding: 0 20rpx;
font-size: 28rpx;
&:focus {
border-color: #ff9a9e;
background: #fff;
}
}
}
}
// 地址表单项特殊样式
.address-form-item {
margin-bottom: 20rpx;
}
// 地址输入框包装器
.address-input-wrapper {
flex: 3;
display: flex;
align-items: center;
height: 80rpx;
padding: 0 20rpx;
border-radius: 12rpx;
background: #fff;
//border: 1rpx solid #e0e0e0;
transition: all 0.3s ease;
&:focus-within {
border-color: #f15a04;
box-shadow: 0 0 0 2rpx rgba(241, 90, 4, 0.1);
}
.address-input {
flex: 1;
height: 100%;
font-size: 28rpx;
color: #333;
border: none;
outline: none;
background: transparent;
&::placeholder {
color: #999;
}
}
.map-icon-wrapper {
display: flex;
align-items: center;
justify-content: center;
width: 48rpx;
height: 48rpx;
background: #f15a04;
border-radius: 50%;
transition: all 0.3s ease;
//box-shadow: 0 4rpx 12rpx rgba(241, 90, 4, 0.3);
margin-left: 16rpx;
&:active {
transform: scale(0.9);
box-shadow: 0 2rpx 8rpx rgba(241, 90, 4, 0.4);
}
.map-icon {
font-size: 28rpx;
color: white;
}
}
}
// 地址选择器(保留原有样式,以防其他地方使用)
.address-selector {
flex: 3;
display: flex;
align-items: center;
justify-content: space-between;
height: 80rpx;
padding: 0 20rpx;
border-radius: 12rpx;
transition: all 0.3s ease;
&:active {
background: #f0f0f0;
border-color: #ff6b6b;
}
.address-text {
flex: 1;
font-size: 28rpx;
color: #666;
margin-right: 20rpx;
}
.map-icon-wrapper {
display: flex;
align-items: center;
justify-content: center;
width: 48rpx;
height: 48rpx;
background: #f15a04;
border-radius: 50%;
transition: all 0.3s ease;
box-shadow: 0 4rpx 12rpx rgba(241, 90, 4, 0.3);
&:active {
transform: scale(0.9);
box-shadow: 0 2rpx 8rpx rgba(241, 90, 4, 0.4);
}
.map-icon {
font-size: 28rpx;
color: white;
}
}
}
// 位置建议卡片
.location-suggestion {
margin-bottom: 40rpx;
.location-card {
display: flex;
align-items: center;
justify-content: space-between;
background: white;
border: 2rpx solid #e8f4fd;
border-radius: 16rpx;
padding: 24rpx;
box-shadow: 0 4rpx 12rpx rgba(0, 0, 0, 0.05);
position: relative;
&::before {
content: '';
position: absolute;
left: 0;
top: 0;
bottom: 0;
width: 6rpx;
background: #f3f5f6;
border-radius: 6rpx 0 0 6rpx;
}
.location-content {
flex: 1;
margin-right: 20rpx;
.location-header {
display: flex;
align-items: center;
margin-bottom: 8rpx;
.location-title {
font-size: 24rpx;
color: #f15a04;
font-weight: 500;
margin-right: 12rpx;
}
.location-company {
font-size: 26rpx;
color: #333;
font-weight: 500;
}
}
.location-address {
font-size: 24rpx;
color: #666;
line-height: 1.4;
}
}
.use-location-btn {
background: #f15a04;
color: #000000;
border: none;
border-radius: 24rpx;
padding: 12rpx 24rpx;
font-size: 24rpx;
font-weight: 500;
box-shadow: 0 4rpx 12rpx rgba(255, 107, 107, 0.3);
transition: all 0.3s ease;
&:active {
transform: translateY(2rpx);
box-shadow: 0 2rpx 8rpx rgba(255, 107, 107, 0.4);
}
}
.get-location-btn {
background: #f15a04;
color: #ffffff;
border: none;
border-radius: 24rpx;
padding: 12rpx 24rpx;
font-size: 24rpx;
font-weight: 500;
box-shadow: 0 4rpx 12rpx rgba(255, 107, 107, 0.3);
transition: all 0.3s ease;
&:active {
transform: translateY(2rpx);
box-shadow: 0 2rpx 8rpx rgba(255, 107, 107, 0.4);
}
}
}
}
// 选择器
.selector {
flex: 3;
display: flex;
justify-content: space-between;
align-items: center;
height: 80rpx;
border-radius: 12rpx;
padding: 0 20rpx;
transition: all 0.3s ease;
&:active {
background: #f0f0f0;
border-color: #ff6b6b;
}
.selector-text {
font-size: 28rpx;
color: #666;
}
.arrow-icon {
font-size: 24rpx;
color: #999;
transition: transform 0.3s ease;
}
&:active .arrow-icon {
transform: scale(1.2);
}
}
// 支付区域
.payment-section {
display: flex;
flex-direction: column;
.pay-button {
width: 100%;
height: 100%;
background: #f15a04;
color: #fff;
border: none;
border-radius: 10rpx;
font-size: 32rpx;
font-weight: bold;
margin-bottom: 40rpx;
box-shadow: 0 10rpx 30rpx rgba(255, 154, 158, 0.3);
}
.payment-details {
.details-header {
display: flex;
align-items: center;
padding: 20rpx 0;
border-bottom: 2rpx solid #f0f0f0;
.details-title {
font-size: 28rpx;
color: #f15a04;
padding-right: 10rpx;
}
.arrow-wrapper {
display: flex;
align-items: center;
justify-content: center;
transition: transform 0.3s ease;
&.expanded {
transform: rotate(180deg);
}
}
.details-arrow {
width: 20rpx;
height: 10rpx;
}
}
.details-content {
padding: 20rpx 0;
.detail-item {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 20rpx;
.detail-label {
font-size: 28rpx;
color: #868686;
}
.detail-value {
font-size: 28rpx;
color: #3d3d3d;
font-weight: bold;
}
}
}
}
}
// 底部导航样式已移除使用系统tabBar
// 动画
@keyframes pulse {
0% {
transform: scale(1);
opacity: 1;
}
50% {
transform: scale(1.1);
opacity: 0.8;
}
100% {
transform: scale(1);
opacity: 1;
}
}
</style>