285 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Vue
		
	
	
	
			
		
		
	
	
			285 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Vue
		
	
	
	
| <template>
 | |
|   <div
 | |
|     v-if="popupVisible"
 | |
|     class="region-select-overlay"
 | |
|     @click.self="handleClose"
 | |
|   >
 | |
|     <div class="region-select-popup">
 | |
|       <div class="popup-header">
 | |
|         <h3 class="popup-title">选择收益区域</h3>
 | |
|       </div>
 | |
|       <div class="popup-content">
 | |
|         <picker-view
 | |
|           v-if="popupVisible"
 | |
|           class="picker-view"
 | |
|           :indicator-style="indicatorStyle"
 | |
|           :value="pickerValue"
 | |
|           @change="handlePickerChange"
 | |
|         >
 | |
|           <picker-view-column>
 | |
|             <div v-for="p in provinces" :key="p.id" class="picker-item">
 | |
|               {{ p.name }}
 | |
|             </div>
 | |
|           </picker-view-column>
 | |
|           <picker-view-column>
 | |
|             <div v-for="c in cities" :key="c.id" class="picker-item">
 | |
|               {{ c.name }}
 | |
|             </div>
 | |
|           </picker-view-column>
 | |
|           <picker-view-column>
 | |
|             <div v-for="d in districts" :key="d.id" class="picker-item">
 | |
|               {{ d.name }}
 | |
|             </div>
 | |
|           </picker-view-column>
 | |
|         </picker-view>
 | |
|       </div>
 | |
|       <div class="popup-footer">
 | |
|         <button class="popup-btn popup-cancel" @click="handleClose">
 | |
|           取消
 | |
|         </button>
 | |
|         <button class="popup-btn popup-confirm" @click="handleConfirm">
 | |
|           确认
 | |
|         </button>
 | |
|       </div>
 | |
|     </div>
 | |
|   </div>
 | |
| </template>
 | |
| 
 | |
| <script>
 | |
| import { getRegionSelect, setRegion, getRegionAreaTree } from '@/config/mine.js'
 | |
| 
 | |
| export default {
 | |
|   name: 'region-select',
 | |
|   data() {
 | |
|     return {
 | |
|       popupVisible: false,
 | |
|       areaTree: [],
 | |
|       provinces: [],
 | |
|       cities: [],
 | |
|       districts: [],
 | |
|       pickerValue: [0, 0, 0],
 | |
|       indicatorStyle: `height: 50px;`,
 | |
|     }
 | |
|   },
 | |
|   async created() {
 | |
|     uni.showLoading({
 | |
|       title: '加载中...',
 | |
|     })
 | |
|     try {
 | |
|       const res = await getRegionSelect()
 | |
|       if (
 | |
|         res.code === 200 &&
 | |
|         res.data &&
 | |
|         res.data.regionStatus === 0 &&
 | |
|         !res.data.province
 | |
|       ) {
 | |
|         this.open()
 | |
|       }
 | |
|     } catch (error) {
 | |
|       console.error('Failed to get region select info:', error)
 | |
|     } finally {
 | |
|       uni.hideLoading()
 | |
|     }
 | |
|   },
 | |
|   methods: {
 | |
|     async open() {
 | |
|       uni.showLoading({
 | |
|         title: '加载中...',
 | |
|       })
 | |
|       return new Promise(async (resolve, reject) => {
 | |
|         if (this.areaTree?.length === 0) {
 | |
|           await this.loadAreaTree()
 | |
|         }
 | |
|         this.popupVisible = true
 | |
|         resolve()
 | |
|         uni.hideLoading()
 | |
|       })
 | |
|     },
 | |
|     async loadAreaTree() {
 | |
|       try {
 | |
|         const res = await getRegionAreaTree()
 | |
|         if (res.code === 200 && res.data) {
 | |
|           this.areaTree = res.data
 | |
|           this.provinces = this.areaTree
 | |
|           if (this.provinces.length > 0) {
 | |
|             this.cities = this.provinces[0].children || []
 | |
|             if (this.cities.length > 0) {
 | |
|               this.districts = this.cities[0].children || []
 | |
|             } else {
 | |
|               this.districts = []
 | |
|             }
 | |
|           }
 | |
|           return true
 | |
|         }
 | |
|         return false
 | |
|       } catch (error) {
 | |
|         console.error('Failed to load area tree:', error)
 | |
|       }
 | |
|     },
 | |
|     handlePickerChange(e) {
 | |
|       const [pIndex, cIndex] = e.detail.value
 | |
|       const oldPIndex = this.pickerValue[0]
 | |
|       const oldCIndex = this.pickerValue[1]
 | |
| 
 | |
|       this.pickerValue = e.detail.value
 | |
| 
 | |
|       if (pIndex !== oldPIndex) {
 | |
|         this.cities = this.provinces[pIndex]?.children || []
 | |
|         this.districts = this.cities[0]?.children || []
 | |
|       } else if (cIndex !== oldCIndex) {
 | |
|         this.districts = this.cities[cIndex]?.children || []
 | |
|       }
 | |
|     },
 | |
|     handleClose() {
 | |
|       this.popupVisible = false
 | |
|     },
 | |
|     async handleConfirm() {
 | |
|       const [pIndex, cIndex, dIndex] = this.pickerValue
 | |
|       const province = this.provinces[pIndex]
 | |
|       const city = this.cities[cIndex]
 | |
|       const district = this.districts[dIndex]
 | |
| 
 | |
|       if (!province || !city || !district) {
 | |
|         return
 | |
|       }
 | |
|       try {
 | |
|         const params = {
 | |
|           province: province.id,
 | |
|           city: city.id,
 | |
|           county: district.id,
 | |
|         }
 | |
|         const res = await setRegion(params)
 | |
|         if (res.code === 200) {
 | |
|           this.handleClose()
 | |
|           this.$emit('success')
 | |
|         } else {
 | |
|           throw new Error(res.message || 'Failed to set region')
 | |
|         }
 | |
|       } catch (error) {
 | |
|         console.error('Failed to set region:', error)
 | |
|       }
 | |
|     },
 | |
|   },
 | |
| }
 | |
| </script>
 | |
| 
 | |
| <style scoped>
 | |
| .region-select-overlay {
 | |
|   position: fixed;
 | |
|   top: 0;
 | |
|   left: 0;
 | |
|   width: 100%;
 | |
|   height: 100%;
 | |
|   background-color: rgba(0, 0, 0, 0.5);
 | |
|   display: flex;
 | |
|   align-items: center;
 | |
|   justify-content: center;
 | |
|   z-index: 1000;
 | |
|   transition: opacity 0.3s ease;
 | |
| }
 | |
| 
 | |
| .region-select-popup {
 | |
|   width: 90%;
 | |
|   max-width: 680rpx;
 | |
|   background-color: white;
 | |
|   border-radius: 24rpx;
 | |
|   animation: scale-up 0.3s ease-out;
 | |
|   max-height: 80vh;
 | |
|   display: flex;
 | |
|   flex-direction: column;
 | |
|   overflow: hidden;
 | |
|   box-shadow: 0 4rpx 24rpx rgba(0, 0, 0, 0.1);
 | |
| }
 | |
| 
 | |
| .popup-header {
 | |
|   display: flex;
 | |
|   justify-content: center;
 | |
|   align-items: center;
 | |
|   padding: 30rpx;
 | |
|   border-bottom: 1rpx solid #f5f5f5;
 | |
|   flex-shrink: 0;
 | |
| }
 | |
| 
 | |
| .popup-title {
 | |
|   font-size: 32rpx;
 | |
|   font-weight: 600;
 | |
|   color: #1c1c1e;
 | |
|   margin: 0;
 | |
| }
 | |
| 
 | |
| .popup-btn {
 | |
|   border: none;
 | |
|   background: none;
 | |
|   cursor: pointer;
 | |
| }
 | |
| 
 | |
| .popup-content {
 | |
|   overflow: hidden;
 | |
|   flex-grow: 1;
 | |
| }
 | |
| 
 | |
| .picker-view {
 | |
|   width: 100%;
 | |
|   height: 500rpx;
 | |
| }
 | |
| 
 | |
| .picker-item {
 | |
|   display: flex;
 | |
|   align-items: center;
 | |
|   justify-content: center;
 | |
|   height: 50px;
 | |
|   font-size: 30rpx;
 | |
|   color: #666;
 | |
|   white-space: nowrap;
 | |
|   overflow: hidden;
 | |
|   text-overflow: ellipsis;
 | |
| }
 | |
| 
 | |
| .popup-footer {
 | |
|   display: flex;
 | |
|   flex-shrink: 0;
 | |
|   padding: 24rpx;
 | |
|   gap: 24rpx;
 | |
|   background-color: #f7f7f7;
 | |
|   border-top: 1rpx solid #efefef;
 | |
| }
 | |
| 
 | |
| .popup-footer .popup-btn {
 | |
|   flex: 1;
 | |
|   text-align: center;
 | |
|   padding: 14rpx 0;
 | |
|   font-size: 28rpx;
 | |
|   border-radius: 40rpx;
 | |
|   font-weight: 500;
 | |
|   transition:
 | |
|     transform 0.1s ease,
 | |
|     box-shadow 0.2s ease;
 | |
| }
 | |
| 
 | |
| .popup-footer .popup-btn:active {
 | |
|   transform: scale(0.97);
 | |
| }
 | |
| 
 | |
| .popup-footer .popup-cancel {
 | |
|   background-color: #eee;
 | |
|   color: #555;
 | |
| }
 | |
| 
 | |
| .popup-footer .popup-confirm {
 | |
|   background: linear-gradient(135deg, #007aff, #0056b3);
 | |
|   color: white;
 | |
|   box-shadow: 0 4rpx 12rpx rgba(0, 122, 255, 0.25);
 | |
| }
 | |
| 
 | |
| @keyframes scale-up {
 | |
|   from {
 | |
|     transform: scale(0.8);
 | |
|     opacity: 0;
 | |
|   }
 | |
|   to {
 | |
|     transform: scale(1);
 | |
|     opacity: 1;
 | |
|   }
 | |
| }
 | |
| </style>
 |