choose-address.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544
  1. <template>
  2. <view class="my-address">
  3. <!-- 定位区 -->
  4. <view class="search-box">
  5. <tui-icon name="arrowleft" :size="25" color="#00" @click="handleBack"></tui-icon>
  6. <view class="search-wrapper">
  7. <tui-icon class="search-icon" name="search" :size="20"></tui-icon>
  8. <input v-model="searchCity" type="text" placeholder="请输入所在城市" />
  9. <tui-icon v-if="searchCity" name="close" :size="20" @click="handleClearSearch"></tui-icon>
  10. </view>
  11. <button v-if="searchCity" class="uni-btn" @click="handleSearchCity">搜索</button>
  12. </view>
  13. <view class="current-address" @click="handleGetCurrentAddress">
  14. <text class="current-address-text">
  15. 当前:{{ $store.getters.currentCity ? $store.getters.currentCity : '定位失败,重新定位' }}
  16. </text>
  17. <view>
  18. <tui-icon :size="16" color="#000" name="location"></tui-icon>
  19. <text>重新定位</text>
  20. </view>
  21. </view>
  22. <!-- 热门城市 -->
  23. <view class="hot-city">
  24. <view class="title">热门城市</view>
  25. <tui-grid unlined>
  26. <block v-for="(item, index) in hotCities" :key="index">
  27. <tui-grid-item :cell="3" @click="confirmChooseAddress(item, true)">
  28. <text class="tui-grid-label">
  29. {{ item.level === 4 ? item.town : item.level === 3 ? item.distinguish : item.city }}
  30. </text>
  31. </tui-grid-item>
  32. </block>
  33. </tui-grid>
  34. </view>
  35. <view v-if="currentTab === 1 && currentTownData" style="text-align: right;">
  36. <tui-button
  37. style="display: inline-block;" margin="0 10rpx 8upx 0" width="120rpx" height="50rpx"
  38. type="warning" shape="circle" @click="confirmChooseAddress({ name: '' })"
  39. >
  40. 选择
  41. </tui-button>
  42. </view>
  43. <!-- tabs 标签页 -->
  44. <tui-tabs
  45. :tabs="tabs" selected-color="#e95d20" slider-bg-color="#e95d20" item-width="30%"
  46. :current-tab="currentTab"
  47. @change="handleChangeTab"
  48. ></tui-tabs>
  49. <!-- 标签页 -->
  50. <view v-if="cityList.length" class="wrapper-container">
  51. <swiper disable-touch :current="currentTab" class="swiper" @change="handleChangeSwiper">
  52. <swiper-item class="" item-id="">
  53. <view class="address-list-wrapper">
  54. <tui-index-list
  55. active-key-color="#e95d20" active-color="#e95d20" active-key-background="#fff"
  56. :list-data="cityList"
  57. >
  58. <template #item="{ entity }">
  59. <tui-list-cell v-for="(item, index) in entity" :key="index" padding="16rpx 30rpx">
  60. <view class="tui-list__item" @click="handleChooseCity(item)">
  61. <view :id="'item' + item.name" class="tui-name">{{ item.name }}</view>
  62. </view>
  63. </tui-list-cell>
  64. </template>
  65. </tui-index-list>
  66. </view>
  67. </swiper-item>
  68. <swiper-item>
  69. <view class="choose-cities">
  70. <tui-grid unlined>
  71. <block v-for="(item, index) in currentDistinguishData" :key="index">
  72. <tui-grid-item
  73. class="grid-item" :class="{
  74. active: tabs[1].name === item.name.slice(0, 3) + '...'
  75. }" :cell="3" @click="handleChooseDistinguish(item)"
  76. >
  77. <text class="tui-grid-label">{{ item.name }}</text>
  78. </tui-grid-item>
  79. </block>
  80. </tui-grid>
  81. </view>
  82. </swiper-item>
  83. <swiper-item>
  84. <view class="choose-cities">
  85. <tui-grid unlined>
  86. <block v-for="(item, index) in currentTownData" :key="index">
  87. <tui-grid-item :cell="3" @click="confirmChooseAddress(item)">
  88. <text class="tui-grid-label">{{ item.name }}</text>
  89. </tui-grid-item>
  90. </block>
  91. </tui-grid>
  92. </view>
  93. </swiper-item>
  94. </swiper>
  95. </view>
  96. <view v-else class="no-data"> 暂无数据~ </view>
  97. <tui-popup
  98. :duration="500" :mode-class="[ 'fade-in' ]" :styles="styles" :show="showAuthPopupVisible"
  99. @click="showAuthPopupVisible = false"
  100. >
  101. <view class="address-text">
  102. <tui-icon name="gps" :size="30" color="#e95d20"></tui-icon>
  103. "团蜂"想访问您的地理位置,将根据你的地理位置提供准确的收货地址,社区服务地址,查看附近商家及门店等功能
  104. </view>
  105. </tui-popup>
  106. <tui-toast ref="toast"></tui-toast>
  107. </view>
  108. </template>
  109. <script>
  110. // import { hotCities } from './data';
  111. const hotCities = [
  112. {
  113. city: '佛山市',
  114. town: '大良街道',
  115. distinguish: '顺德区',
  116. level: 4
  117. },
  118. {
  119. city: '佛山市',
  120. town: '龙江镇',
  121. distinguish: '顺德区',
  122. level: 4
  123. },
  124. {
  125. city: '佛山市',
  126. town: '容桂街道',
  127. distinguish: '顺德区',
  128. level: 4
  129. },
  130. {
  131. city: '佛山市',
  132. town: '乐从镇',
  133. distinguish: '顺德区',
  134. level: 4
  135. },
  136. {
  137. city: '贵港市',
  138. town: '',
  139. distinguish: '港北区',
  140. level: 3
  141. }
  142. ]
  143. export default {
  144. name: 'ChooseAddress',
  145. data() {
  146. return {
  147. currentTab: 0,
  148. cityList: [],
  149. searchValue: '',
  150. isShowLoading: true,
  151. showAuthPopupVisible: false,
  152. styles: {
  153. 'position': 'fixed',
  154. 'bottom': 0,
  155. 'top': 0,
  156. 'left': 0,
  157. 'right': 0,
  158. 'display': 'flex',
  159. 'justify-content': 'center',
  160. 'align-items': 'flex-start',
  161. 'background-color': 'rgba(0, 0, 0, 0.5)',
  162. 'padding': '50rpx 0 0 0'
  163. },
  164. tabs: [
  165. {
  166. name: '所在城市'
  167. },
  168. {
  169. name: '区/县'
  170. },
  171. {
  172. name: '镇/街道'
  173. }
  174. ],
  175. mainHeight: 0,
  176. currentDistinguishData: null,
  177. currentTownData: null,
  178. searchCity: '',
  179. allCityData: {},
  180. hotCities: Object.freeze(hotCities),
  181. backUrl: null,
  182. eventName: ''
  183. }
  184. },
  185. watch: {
  186. searchCity(val) {
  187. if (!val) {
  188. this.currentTab = 0
  189. this.currentDistinguishData = null
  190. this.currentTownData = null
  191. this.cityList = this.allCityData
  192. }
  193. }
  194. },
  195. mounted() {
  196. this.getData()
  197. },
  198. methods: {
  199. getData() {
  200. uni.showLoading()
  201. const _this = this
  202. import('./cities.json').then((res) => {
  203. for (const key in res) {
  204. _this.cityList.push(res[key])
  205. }
  206. _this.allCityData = Object.freeze(_this.cityList)
  207. _this.isShowLoading = false
  208. uni.hideLoading()
  209. })
  210. },
  211. changeTab(e) {
  212. this.currentTab = e.index
  213. },
  214. handleSearchCity() {
  215. this.currentTab = 0
  216. this.currentDistinguishData = null
  217. this.currentTownData = null
  218. let data = JSON.parse(JSON.stringify(this.allCityData))
  219. data = data.filter((item) => {
  220. item.data = item.data && item.data.filter((cities) => cities.name.includes(this.searchCity))
  221. return item.data && item.data.length
  222. })
  223. this.cityList = data
  224. },
  225. handleBack(mean) {
  226. if (this.backUrl) {
  227. this.$switchTab('/')
  228. return
  229. }
  230. uni.navigateBack({
  231. delta: 1,
  232. success: mean === 'success' && this.eventName ? uni.$emit(this.eventName) : () => { }
  233. })
  234. },
  235. handleClearSearch() {
  236. this.searchCity = ''
  237. },
  238. handleChooseCity(chooseAddressInfo) {
  239. this.currentDistinguishData = chooseAddressInfo.children
  240. this.currentTownData = null
  241. this.currentTab = 1
  242. this.tabs[0].name = chooseAddressInfo.name.slice(0, 3) + '...'
  243. this.tabs[0].select = chooseAddressInfo.name
  244. // this.$store.commit('location/CHANGE_CURRENT_CITY', cityName)
  245. // this.handleBack()
  246. },
  247. // 滑动swiper
  248. handleChangeSwiper(e) {
  249. const nextIndex = e.detail.current
  250. if (nextIndex === 1 && !this.currentDistinguishData) {
  251. this.currentTab = nextIndex - 1
  252. return
  253. }
  254. this.currentTab = nextIndex
  255. },
  256. // 开始定位
  257. handleGetCurrentAddress() {
  258. // #ifdef APP
  259. const appAuthorizeSetting = uni.getAppAuthorizeSetting()
  260. if (appAuthorizeSetting.locationAuthorized !== 'authorized') {
  261. this.showAuthPopupVisible = true
  262. this.$store.dispatch('location/getCurrentLocation')
  263. } else {
  264. this.$store.dispatch('location/getCurrentLocation')
  265. }
  266. // #endif
  267. // #ifdef H5
  268. this.$store.dispatch('location/getCurrentLocation')
  269. // #endif
  270. },
  271. // 切换tab
  272. handleChangeTab(info) {
  273. if (info.index === 1 && !this.currentDistinguishData) {
  274. this.ttoast({
  275. type: 'fail',
  276. title: '请选择所在城市'
  277. })
  278. return
  279. }
  280. if (info.index === 2 && !this.currentTownData) {
  281. this.ttoast({
  282. type: 'fail',
  283. title: '请选择所在区县'
  284. })
  285. return
  286. }
  287. this.currentTab = info.index
  288. },
  289. // 选择区县
  290. handleChooseDistinguish(data) {
  291. this.currentTownData = data.children
  292. // this.currentTab = 2
  293. this.tabs[1].name = data.name.slice(0, 3) + '...'
  294. this.tabs[1].select = data.name
  295. },
  296. // 选择定位
  297. async confirmChooseAddress(data, isHot) {
  298. uni.showLoading()
  299. if (isHot) {
  300. await this.$store.dispatch('location/getDetailAddress', data)
  301. } else {
  302. // console.log(this.tabs)
  303. await this.$store.dispatch('location/getDetailAddress', {
  304. city: this.tabs[0].select,
  305. distinguish: this.tabs[1].select,
  306. town: data.name
  307. })
  308. }
  309. uni.hideLoading()
  310. this.ttoast('修改成功')
  311. setTimeout(() => {
  312. if (this.backUrl) {
  313. uni.redirectTo({
  314. url: this.backUrl
  315. })
  316. } else {
  317. this.handleBack('success')
  318. }
  319. }, 1000)
  320. }
  321. },
  322. onLoad(params) {
  323. this.eventName = params.eventName || ''
  324. if (params.backUrl) {
  325. this.backUrl = params.backUrl.replaceAll('_', '?').replaceAll('|', '/')
  326. }
  327. }
  328. }
  329. </script>
  330. <style lang="scss" scoped>
  331. .my-address {
  332. width: 100vw;
  333. min-height: 100vh;
  334. background-color: #f3f3f3;
  335. .search-box {
  336. display: flex;
  337. align-items: center;
  338. justify-content: space-between;
  339. width: 100%;
  340. height: 100upx;
  341. padding: 20upx;
  342. box-sizing: border-box;
  343. background-color: #fff;
  344. .search-wrapper {
  345. height: 100%;
  346. background-color: #f3f3f3;
  347. border-radius: 100px;
  348. display: flex;
  349. align-items: center;
  350. flex: 1;
  351. padding: 0 20upx;
  352. input {
  353. flex: 1;
  354. font-size: 24upx;
  355. }
  356. .search-icon {
  357. padding-right: 16upx;
  358. margin-right: 16upx !important;
  359. border-right: 1upx solid #d2d2d2;
  360. }
  361. }
  362. }
  363. .current-address {
  364. display: flex;
  365. align-items: center;
  366. justify-content: space-between;
  367. padding: 20upx;
  368. box-sizing: border-box;
  369. background-color: #fff;
  370. background-color: #fff;
  371. font-size: 24upx;
  372. margin-bottom: 40upx;
  373. .current-address-text {
  374. display: flex;
  375. align-items: center;
  376. width: 300upx;
  377. overflow: hidden;
  378. white-space: nowrap;
  379. text-overflow: ellipsis;
  380. }
  381. view {
  382. display: flex;
  383. align-items: center;
  384. text {
  385. margin-left: 10upx;
  386. }
  387. }
  388. }
  389. .hot-city {
  390. background-color: #fff;
  391. margin-bottom: 30upx;
  392. .title {
  393. width: 100%;
  394. position: relative;
  395. display: flex;
  396. align-items: center;
  397. background: #ffffff;
  398. color: rgb(102, 102, 102);
  399. font-size: 13px;
  400. height: 30px;
  401. padding: 0px 10px;
  402. font-weight: bold;
  403. font-size: 26upx;
  404. }
  405. .city-list {
  406. display: flex;
  407. align-items: center;
  408. flex-wrap: wrap;
  409. padding: 20upx;
  410. box-sizing: border-box;
  411. padding-left: 40upx;
  412. .hot-city-item {
  413. font-size: 24upx;
  414. color: #000;
  415. padding: 10upx 20upx;
  416. border: 1upx solid #adadad;
  417. margin-right: 30upx;
  418. margin-bottom: 20upx;
  419. border-radius: 10upx;
  420. transition: all 350ms;
  421. &:active {
  422. background-color: #e8e8e8;
  423. }
  424. }
  425. }
  426. }
  427. /deep/ .tui-title__item {
  428. background-color: #fff !important;
  429. }
  430. }
  431. .address-text {
  432. display: flex;
  433. align-items: center;
  434. justify-content: center;
  435. width: 600upx;
  436. padding: 26upx;
  437. background-color: #fff;
  438. border-radius: 20upx;
  439. font-size: 32upx;
  440. line-height: 1.5;
  441. /deep/ .tui-icon {
  442. margin-right: 10upx !important;
  443. }
  444. }
  445. .uni-btn {
  446. font-size: 28upx;
  447. margin-left: 10upx;
  448. color: rgb(233, 93, 32);
  449. }
  450. .no-data {
  451. height: 300upx;
  452. text-align: center;
  453. line-height: 400upx;
  454. color: #ccc;
  455. font-size: 28upx;
  456. }
  457. .wrapper-container {
  458. width: 100%;
  459. height: calc(100vh - 292upx);
  460. // background-color: #f40;
  461. overflow: hidden;
  462. .swiper {
  463. height: calc(100vh - 292upx);
  464. }
  465. /deep/ .tui-scroll__view {
  466. height: calc(100vh - 292upx) !important;
  467. }
  468. }
  469. .choose-cities {
  470. width: 100%;
  471. height: 100%;
  472. }
  473. /deep/ .tui-tabs-item {
  474. width: 160rpx;
  475. white-space: nowrap;
  476. overflow: hidden;
  477. text-overflow: ellipsis;
  478. }
  479. /deep/ .tui-grid {
  480. text-align: center;
  481. }
  482. .grid-item.active {
  483. background-color: #e95d20 !important;
  484. color: #fff;
  485. }
  486. .tui-grid-label {
  487. font-size: 28upx;
  488. width: 100px;
  489. overflow: hidden;
  490. text-overflow: ellipsis;
  491. white-space: nowrap;
  492. display: inline-block;
  493. }
  494. </style>