chose-city.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677
  1. <template>
  2. <view class="chose-city">
  3. <view>
  4. <view class="status_bar" :style="{height: iStatusBarHeight + 'px'}" v-if="topshow"></view>
  5. <view class="nav" v-if="topshow">城市选择
  6. <image src="../../static/img/goback.png" mode="" class="foback" @click="closeModal"></image>
  7. </view>
  8. </view>
  9. <!-- 城市搜索 -->
  10. <view class="city-search-wrap">
  11. <view class="search">
  12. <view class="l-search">
  13. <view class="icon-search">
  14. <!-- <view class="cuIcon-search"></view> -->
  15. <image src="../../static/img/icon_search.png" mode="" class="cuIcon-search"></image>
  16. </view>
  17. <input class="input-search" type="text" :value="inputValue" placeholder="请输入城市" placeholder-style="color:#8E8F97"
  18. :focus="searchFocus" @input="searchChange" />
  19. <text class="clear-input iconfont icon-icon-test" v-if="isClearBtn" @click="inputValue = ''"></text>
  20. </view>
  21. <view class="r-cancel" @click="closeModal">取消</view>
  22. </view>
  23. <!-- 搜索列表 -->
  24. <view class="reach-content" v-show="inputValue">
  25. <block v-show="searchData.length">
  26. <view class="li" v-for="item in searchData" :key="item.citycode" @click="selectCity(item)">
  27. {{item.name}}
  28. </view>
  29. </block>
  30. <view class="has-no-data" v-show="hasNoData">没有找到匹配数据~</view>
  31. </view>
  32. </view>
  33. <!-- 城市列表 -->
  34. <scroll-view class="scroll-view" scroll-y scroll-with-animation="true" enable-back-to-top="true" :scroll-into-view="toIndex"
  35. @scroll="scrollHandle" v-if="!inputValue">
  36. <view class="block">
  37. <!-- 您所在的地区 -->
  38. <view class="area list-item" id="area">
  39. <view class="title-wrapp">
  40. <view class="c-title">
  41. <text class="l">定位城市</text>
  42. </view>
  43. </view>
  44. <view class="ul">
  45. <!-- <view class="li now font-clamp" @click="selectCity(myCityObj,'refresh')">
  46. <image src="https://fulu-mall.oss-cn-hangzhou.aliyuncs.com/e0e5e497b1714e8fa6135df32e4bf114.png" class="icon"
  47. v-if="hasLocation"></image>
  48. <image src="https://fulu-mall.oss-cn-hangzhou.aliyuncs.com/8ed89f5bcaac48c6bd9139261f86c6b6.png" class="icon"
  49. v-else></image>
  50. <text class="text">{{ hasLocation ? myCityObj.name:'定位失败' }}</text>
  51. </view> -->
  52. <view @click="selectCity(citydw)">{{citydw.name}}</view>
  53. </view>
  54. </view>
  55. <!-- 历史记录 -->
  56. <!-- <view class="area list-item" id="record" v-if="recordList.length">
  57. <view class="title-wrapp">
  58. <view class="c-title">
  59. <text class="l">历史记录</text>
  60. </view>
  61. </view>
  62. <view class="ul">
  63. <view class="li font-clamp" v-for="item in recordList" :key="item.citycode" @click="selectCity(item)">
  64. {{ item.name }}
  65. </view>
  66. </view>
  67. </view> -->
  68. </view>
  69. <!-- 城市列表 -->
  70. <view class="city-list">
  71. <view class="list list-item" v-for="(item, key) of cityList" :key="key" :id="item.nameType">
  72. <view class="c-title">{{ item.nameType }}</view>
  73. <view class="item" v-for="innerItem in item.list" :key="innerItem.citycode" @click="selectCity(innerItem)">
  74. {{ innerItem.name }}
  75. </view>
  76. </view>
  77. </view>
  78. </scroll-view>
  79. <!-- 字母列表 -->
  80. <view class="alphabet" @touchstart="touchStart" @touchend="touchEnd" @touchmove.stop="touchMove">
  81. <view v-for="(item, index) in alphabet" :key="index" @touchstart="getLetter" @touchend="setLetter" :id="item">
  82. <view class="item" :class="{ active: currentLetter == item }">
  83. {{ item == 'area' ? '当前' : item == 'record' ? '历史' : item }}
  84. </view>
  85. </view>
  86. </view>
  87. </view>
  88. </template>
  89. <script>
  90. import cityJson from '@/static/dataJson/city.json'
  91. export default {
  92. props:{
  93. topshow:{ //参数名
  94. type:Boolean, //定义传值的类型
  95. default:false //参数默认
  96. }
  97. },
  98. data() {
  99. return {
  100. isIPX: null,
  101. regionId: null, // 区域ID
  102. isToggle: true,
  103. isReach: false,
  104. inputValue: '',
  105. searchData: [], // 搜索的数据
  106. isClearBtn: false,
  107. toIndex: '', // 跳转的索引的字母
  108. tipsLetter: '', // 滑动显示字母
  109. timer: null,
  110. hasNoData: false,
  111. searchFocus: false,
  112. letterDetails: [],
  113. currentLetter: 'area', //默认选择
  114. cityArr: [],
  115. recordList: [],
  116. cityList: [],
  117. hasLocation: false,
  118. myCityObj: {},
  119. alphabet:[],
  120. iStatusBarHeight:'',
  121. lng:'',
  122. lat:'',
  123. cityname:'',
  124. cityCode:'',
  125. citydw:{
  126. name:'',
  127. citycode:'',
  128. }
  129. };
  130. },
  131. mounted() {
  132. this.iStatusBarHeight = uni.getSystemInfoSync().statusBarHeight;
  133. this.cityArr = cityJson.data.list
  134. if (this.cityArr && this.cityArr[0]) {
  135. this.cityArr.map(v => {
  136. v.nameType = v.pinyin.substr(0, 1)
  137. })
  138. this.cityList = this.groupArr(this.cityArr, 'nameType')
  139. }
  140. this.recordList = cityJson.data.recordList
  141. this.alphabet = cityJson.data.alphabet
  142. this.getLocation()
  143. },
  144. watch: {
  145. // 城市搜索输入框
  146. inputValue(newVal) {
  147. this.isClearBtn = newVal ? true : false;
  148. if (this.timer) {
  149. clearTimeout(this.timer);
  150. }
  151. if (!this.inputValue) {
  152. this.searchData = [];
  153. return;
  154. }
  155. this.timer = setTimeout(() => {
  156. const result = [];
  157. this.cityList.map(v => {
  158. v.list.forEach((item) => {
  159. if (/^[a-zA-Z]+$/.test(item.pinyin) && item.pinyin.toLowerCase().includes(this.inputValue.toLowerCase()) ||
  160. item.name.includes(this.inputValue)) {
  161. result.push(item);
  162. }
  163. });
  164. })
  165. this.searchData = result;
  166. if (this.searchData.length === 0) {
  167. this.hasNoData = true;
  168. } else {
  169. this.hasNoData = false;
  170. }
  171. }, 500);
  172. },
  173. isReach(val) {
  174. this.searchFocus = val;
  175. },
  176. },
  177. methods: {
  178. getLocation() {
  179. const that = this
  180. /* uni.getLocation({
  181. // #ifdef MP-ALIPAY
  182. type: 'gcj02',
  183. // #endif
  184. success(res) {
  185. console.log('---',res)
  186. res.name = res.city
  187. res.citycode = res.cityAdcode
  188. that.myCityObj = res
  189. that.hasLocation = true
  190. uni.setStorageSync('nowCityObj', res)
  191. },
  192. fail(err) {
  193. that.hasLocation = false
  194. uni.showToast({
  195. icon:'none',
  196. title: '获取用户定位失败,请手动选择当前城市'
  197. })
  198. },
  199. }) */
  200. console.log("定位")
  201. uni.getLocation({
  202. type: 'gcj02',
  203. success: function (res) {
  204. console.log(res)
  205. that.lng=res.longitude
  206. that.lat=res.latitude
  207. that.getAdress();
  208. },
  209. fail(err) {
  210. console.log(err)
  211. that.cityname=uni.getStorageSync("location").cityname;
  212. that.citydw.name=uni.getStorageSync("location").cityname
  213. that.citydw.citycode=uni.getStorageSync("location").cityCode
  214. }
  215. });
  216. },
  217. getAdress(){
  218. var location=this.lng+','+this.lat
  219. uni.request({
  220. url: 'https://restapi.amap.com/v3/place/around', //仅为示例,并非真实接口地址。
  221. data: {
  222. key: '064b6a4a8ade55656edcde2f528876de',
  223. location: location,
  224. types:"190000",
  225. extensions:"all",
  226. radius:100
  227. },
  228. dataType: "json",
  229. success: (res) => {
  230. console.log(res);
  231. this.cityname=res.data.pois[0].cityname;
  232. var cityCode=res.data.pois[0].adcode
  233. cityCode=cityCode.slice(0, -2)
  234. cityCode=cityCode+'00'
  235. this.cityCode=cityCode
  236. this.citydw.name=this.cityname
  237. this.citydw.citycode=this.cityCode
  238. }
  239. });
  240. },
  241. groupArr(list, field) {
  242. var fieldList = [],
  243. att = [];
  244. list.map((e) => {
  245. fieldList.push(e[field])
  246. })
  247. //数组去重
  248. fieldList = fieldList.filter((e, i, self) => {
  249. return self.indexOf(e) == i
  250. })
  251. for (var j = 0; j < fieldList.length; j++) {
  252. //过滤出匹配到的数据
  253. var arr = list.filter((e) => {
  254. return e[field] == fieldList[j];
  255. })
  256. att.push({
  257. nameType: arr[0].nameType,
  258. list: arr
  259. })
  260. }
  261. return att;
  262. },
  263. selectCity(item,type) {
  264. if(type === 'refresh' && !this.hasLocation){
  265. // 获取定位
  266. return this.getLocation()
  267. }
  268. // console.log('选择的城市:', item);
  269. uni.setStorageSync('myCityObj', item)
  270. this.$emit('selectCity', item)
  271. // 当前项目是需要选择到区域,所以选择城市后回到区县的地方
  272. this.toIndex = 'area';
  273. setTimeout(() => {
  274. this.toIndex = '';
  275. }, 1000);
  276. },
  277. closeModal(){
  278. this.$emit('closeModal')
  279. },
  280. //列表滚动,和右边字母表对应
  281. scrollHandle(e) {
  282. let view = uni.createSelectorQuery().in(this).selectAll('.list-item');
  283. view
  284. .boundingClientRect((d) => {
  285. let top = d[0].top;
  286. d.forEach((item) => {
  287. item.top = item.top - top;
  288. item.bottom = item.bottom - top;
  289. this.letterDetails.push({
  290. id: item.id,
  291. top: item.top,
  292. bottom: item.bottom,
  293. });
  294. });
  295. })
  296. .exec();
  297. const scrollTop = e.detail.scrollTop;
  298. this.letterDetails.some((item) => {
  299. if (scrollTop >= item.top && scrollTop <= item.bottom - 20) {
  300. this.currentLetter = item.id;
  301. //当前固定用的是粘性定位,如果不用粘性定位,在这里设置
  302. return true;
  303. }
  304. });
  305. },
  306. //搜索
  307. searchChange(e) {
  308. let {
  309. value
  310. } = e.detail;
  311. this.inputValue = value;
  312. },
  313. // 触发开始
  314. touchStart(e) {
  315. // console.log(e);
  316. },
  317. //移动时
  318. touchMove(e) {
  319. uni.vibrateShort();
  320. let y = e.touches[0].clientY;
  321. let offsettop = e.currentTarget.offsetTop;
  322. //判断选择区域,只在选择区才会生效
  323. if (y > offsettop) {
  324. let num = parseInt((y - offsettop) / 15); //右边每个字母元素的高度
  325. let letter = this.alphabet[num];
  326. this.tipsLetter = letter;
  327. let curentLetter = this.letterTransform(letter);
  328. uni.showToast({
  329. title: curentLetter,
  330. icon: 'none',
  331. });
  332. }
  333. },
  334. //触发结束
  335. touchEnd() {
  336. this.toIndex = this.tipsLetter;
  337. },
  338. //移动开始获取字母,并放大提示
  339. getLetter(e) {
  340. uni.vibrateShort();
  341. let {
  342. id
  343. } = e.currentTarget;
  344. this.tipsLetter = id;
  345. let curentLetter = this.letterTransform(id);
  346. uni.showToast({
  347. title: curentLetter,
  348. icon: 'none',
  349. });
  350. },
  351. //移动结束设置字母,赋值到toIndex
  352. setLetter() {
  353. this.toIndex = this.tipsLetter;
  354. },
  355. //提示字母转换
  356. letterTransform(letter) {
  357. let str = '';
  358. if (letter == 'area') {
  359. str = '当前';
  360. } else if (letter == 'record') {
  361. str = '历史';
  362. } else {
  363. str = letter;
  364. }
  365. return str;
  366. },
  367. },
  368. }
  369. </script>
  370. <style lang="less" scoped>
  371. .chose-city {
  372. position: fixed;
  373. top: 0;
  374. left: 0;
  375. right: 0;
  376. bottom: 0;
  377. z-index: 999;
  378. background: #fff;
  379. }
  380. .city-search-wrap {
  381. width: 100%;
  382. box-sizing: border-box;
  383. .search {
  384. width: 750rpx;
  385. height: 110rpx;
  386. display: flex;
  387. align-items: center;
  388. font-size: 28rpx;
  389. color: #222;
  390. padding: 14rpx 36rpx;
  391. box-sizing: border-box;
  392. background: #fff;
  393. .l-search {
  394. width: 597rpx;
  395. position: relative;
  396. height: 72rpx;
  397. line-height: 72rpx;
  398. .icon-search {
  399. font-size: 28rpx;
  400. position: absolute;
  401. left: 0rpx;
  402. top: 0;
  403. color: #8e8f97;
  404. font-weight: 700;
  405. height: 72rpx;
  406. line-height: 72rpx;
  407. }
  408. .cuIcon-search{
  409. width: 40rpx;
  410. height: 40rpx;
  411. padding: 20rpx;
  412. }
  413. .input-search {
  414. width: 597rpx;
  415. height: 72rpx;
  416. box-sizing: border-box;
  417. padding: 0 84rpx 0 84rpx;
  418. text-align: left;
  419. background: #f4f5f9;
  420. border-radius: 12rpx;
  421. border: 0;
  422. }
  423. .clear-input {
  424. font-size: 30rpx;
  425. position: absolute;
  426. right: 10rpx;
  427. top: 50%;
  428. transform: translateY(-50%);
  429. padding: 10rpx;
  430. color: #8e8f97;
  431. }
  432. }
  433. .r-cancel {
  434. width: 80rpx;
  435. box-sizing: border-box;
  436. padding-left: 24rpx;
  437. font-size: 28rpx;
  438. height: 72rpx;
  439. line-height: 72rpx;
  440. background: transparent;
  441. border: 0;
  442. color: #519AD2;
  443. }
  444. }
  445. }
  446. .reach-content {
  447. padding-left: 36rpx;
  448. box-sizing: border-box;
  449. .li {
  450. width: 714rpx;
  451. font-size: 28rpx;
  452. height: 100rpx;
  453. line-height: 100rpx;
  454. color: #333;
  455. position: relative;
  456. box-sizing: border-box;
  457. border-bottom: 2rpx solid #F5F5F5;
  458. }
  459. }
  460. .block {
  461. padding: 0 36rpx;
  462. box-sizing: border-box;
  463. }
  464. .top-search {
  465. line-height: 72rpx;
  466. padding: 14rpx 30rpx 0;
  467. box-sizing: border-box;
  468. margin-bottom: 26rpx;
  469. .item {
  470. background: #F5F5F5;
  471. border-radius: 12rpx;
  472. font-size: 28rpx;
  473. text-align: center;
  474. color: #999999;
  475. /* #ifdef MP-ALIPAY */
  476. height: 72rpx;
  477. line-height: 72rpx;
  478. /* #endif */
  479. text {
  480. padding-left: 20rpx;
  481. color: #c1c2cd;
  482. vertical-align: middle;
  483. position: relative;
  484. top: -4rpx;
  485. }
  486. }
  487. }
  488. .scroll-view {
  489. width: 100%;
  490. height: calc(100vh - 110rpx);
  491. box-sizing: border-box;
  492. }
  493. .area {
  494. margin-bottom: 8rpx;
  495. .title-wrapp {
  496. position: sticky;
  497. top: 0;
  498. left: 0;
  499. background: #fff;
  500. }
  501. .c-title {
  502. width: 100%;
  503. box-sizing: border-box;
  504. font-size: 28rpx;
  505. color: #999999;
  506. margin-bottom: 24rpx;
  507. display: inline-flex;
  508. justify-content: space-between;
  509. align-items: center;
  510. .r {
  511. font-size: 24rpx;
  512. color: #8e8f97;
  513. display: inline-block;
  514. align-items: center;
  515. .iconfont {
  516. font-size: 24rpx;
  517. }
  518. }
  519. }
  520. .ul {
  521. display: flex;
  522. flex-wrap: wrap;
  523. .li {
  524. width: 155rpx;
  525. padding: 0 10rpx;
  526. box-sizing: border-box;
  527. height: 72rpx;
  528. line-height: 68rpx;
  529. text-align: center;
  530. font-size: 32rpx;
  531. color: #333;
  532. border-radius: 8rpx;
  533. margin: 0 18rpx 28rpx 0;
  534. border: 2rpx solid #E2E2E2;
  535. &:nth-child(4n) {
  536. margin-right: 0;
  537. }
  538. &.now {
  539. width: auto;
  540. padding: 0 32rpx 0 22rpx;
  541. .icon {
  542. width: 50rpx;
  543. height: 50rpx;
  544. background-size: 100%;
  545. vertical-align: middle;
  546. position: relative;
  547. top: -4rpx;
  548. }
  549. .text {
  550. padding-left: 10rpx;
  551. }
  552. }
  553. &.active {
  554. font-weight: 500;
  555. background: #ffde45;
  556. }
  557. }
  558. .hover {
  559. background: #ffde45;
  560. }
  561. }
  562. }
  563. .city-list {
  564. width: 750rpx;
  565. padding-bottom: 50rpx;
  566. .c-title {
  567. height: 60rpx;
  568. line-height: 60rpx;
  569. font-size: 30rpx;
  570. font-weight: 500;
  571. color: #272636;
  572. background: #fff;
  573. box-sizing: border-box;
  574. padding-left: 36rpx;
  575. position: sticky;
  576. top: 0;
  577. left: 0;
  578. z-index: 2;
  579. }
  580. .item {
  581. width: 714rpx;
  582. margin-left: 36rpx;
  583. padding: 0 36rpx 0 0;
  584. height: 100rpx;
  585. line-height: 100rpx;
  586. color: #333;
  587. font-size: 28rpx;
  588. box-sizing: border-box;
  589. border-bottom: 2rpx solid #F5F5F5;
  590. }
  591. }
  592. .alphabet {
  593. position: fixed;
  594. right: 0;
  595. bottom: 20%;
  596. width: calc(750rpx - 680rpx);
  597. text-align: center;
  598. font-size: 20rpx;
  599. font-weight: 700;
  600. color: #8e8f97;
  601. z-index: 99;
  602. .item {
  603. height: 15px;
  604. line-height: 15px;
  605. }
  606. .active {
  607. color: #222;
  608. }
  609. }
  610. .has-no-data {
  611. font-size: 24rpx;
  612. text-align: center;
  613. color: #8e8f97;
  614. margin-top: 50rpx;
  615. }
  616. .nav{
  617. height: 44px;
  618. line-height: 44px;
  619. text-align: center;
  620. font-size: 26rpx;
  621. color:#3C3C3C;
  622. position: relative;
  623. }
  624. .foback{
  625. position: absolute;
  626. width: 15px;
  627. height: 15px;
  628. left: 24rpx;
  629. top: 15px;
  630. }
  631. </style>