selectCity.vue 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. <template>
  2. <view class="">
  3. <u-popup
  4. :show="show"
  5. mode="bottom"
  6. @close="$emit('close')"
  7. :round="10"
  8. :closeable="true"
  9. >
  10. <view
  11. style="
  12. height: 100rpx;
  13. line-height: 100rpx;
  14. text-align: center;
  15. font-size: 36rpx;
  16. color: #222;
  17. "
  18. >
  19. 选择省市区
  20. </view>
  21. <view>
  22. <input type="text" class="inp" disabled />
  23. <view class="title"> 热门城市 </view>
  24. <view class="city-list">
  25. <view class="city" v-for="(item, index) in hotCityList" :key="index">
  26. {{ item }}
  27. </view>
  28. </view>
  29. <picker-view
  30. class="picker-view"
  31. :indicator-style="indicatorStyle"
  32. :value="value"
  33. @change="bindChange"
  34. :mask-top-style="maskTopStyle"
  35. :mask-bottom-style="maskBottomStyle"
  36. >
  37. <picker-view-column class="picker-view-column">
  38. <view class="item" v-for="(item, index) in cityList" :key="index"
  39. ><text class="text">{{ item.name }}</text></view
  40. >
  41. </picker-view-column>
  42. <picker-view-column class="picker-view-column">
  43. <view class="item" v-for="(item, index) in marketList" :key="index"
  44. ><text class="text">{{ item.name }}</text>
  45. </view>
  46. </picker-view-column>
  47. <picker-view-column class="picker-view-column">
  48. <view
  49. class="item"
  50. v-for="(item, index) in distinguishList"
  51. :key="index"
  52. ><text class="text">{{ item.name }}</text>
  53. </view>
  54. </picker-view-column>
  55. </picker-view>
  56. <button @click="submit" class="confirm">确定</button>
  57. </view>
  58. </u-popup>
  59. </view>
  60. </template>
  61. <script>
  62. export default {
  63. props: {
  64. show: {
  65. typeof: Boolean,
  66. default: false,
  67. },
  68. cityId: {
  69. typeof: Array,
  70. default: () => {
  71. return [];
  72. },
  73. },
  74. countryId: {
  75. typeof: String,
  76. default: "100000",
  77. },
  78. },
  79. watch: {
  80. cityId(newVal) {
  81. if (newVal.length == 3) {
  82. this.cityList.forEach((item, index) => {
  83. if (item.pid == newVal[0]) {
  84. this.value.push(index);
  85. item.children.forEach((items, indexs) => {
  86. if (items.pid == newVal[1]) {
  87. this.value.push(indexs);
  88. items.children.forEach((elem, i) => {
  89. if (elem.pid == newVal[2]) {
  90. this.value.push(i);
  91. }
  92. });
  93. }
  94. });
  95. }
  96. });
  97. } else {
  98. this.value = [0, 0, 0];
  99. }
  100. },
  101. countryId(newVal) {
  102. if (newVal != "100000") {
  103. this.getCityList();
  104. }
  105. },
  106. },
  107. data() {
  108. return {
  109. hotCityList: [
  110. "北京",
  111. "上海",
  112. "深圳",
  113. "广州",
  114. "天津",
  115. "青岛",
  116. "长沙",
  117. "成都",
  118. ],
  119. cityList: [],
  120. title: "picker-view",
  121. marketList: [],
  122. distinguishList: [],
  123. value: [0, 0, 0],
  124. result: [],
  125. indicatorStyle: "height: 50px;",
  126. maskTopStyle: "",
  127. maskBottomStyle: "",
  128. economize: {},
  129. market: {},
  130. distinguish: {},
  131. isValue: true,
  132. };
  133. },
  134. mounted() {
  135. this.getCityList();
  136. },
  137. methods: {
  138. // open() {
  139. // this.getCityList();
  140. // },
  141. //当选中的城市发生变化时调用
  142. bindChange(e) {
  143. this.isValue = false;
  144. const val = e.detail.value;
  145. this.value = val;
  146. console.log(val);
  147. this.marketList = this.cityList[val[0]].children;
  148. this.distinguishList = this.marketList[val[1] || 0].children;
  149. this.economize = this.cityList[val[0]];
  150. this.market = this.marketList[val[1] || 0];
  151. this.distinguish = this.distinguishList[val[2] || 0];
  152. },
  153. //获取城市
  154. getCityList() {
  155. uni.$u.http.get(`/api/area/tree?pid=${this.countryId}`).then((res) => {
  156. this.cityList = res;
  157. if (this.countryId == "100000") {
  158. this.marketList = res[this.value[1]].children;
  159. this.distinguishList = this.marketList[this.value[2]].children;
  160. }
  161. });
  162. },
  163. //提交选中的城市
  164. submit() {
  165. console.log(this.economize);
  166. if (this.isValue) {
  167. console.log(this.value);
  168. this.marketList = this.cityList[this.value[0]].children;
  169. this.distinguishList = this.marketList[this.value[1] || 0].children;
  170. this.economize = this.cityList[this.value[0]];
  171. this.market = this.marketList[this.value[1] || 0];
  172. this.distinguish = this.distinguishList[this.value[2] || 0];
  173. }
  174. const obj = {
  175. ...this.economize,
  176. children: {
  177. ...this.market,
  178. children: this.distinguish,
  179. },
  180. };
  181. this.$emit("close", obj);
  182. },
  183. },
  184. };
  185. </script>
  186. <style lang="scss" scoped>
  187. .title {
  188. margin: 32rpx 26rpx;
  189. }
  190. .inp {
  191. background-color: #f4f4f4;
  192. border-radius: 36rpx;
  193. height: 72rpx;
  194. width: 694rpx;
  195. margin: 30rpx auto 0;
  196. padding-left: 20rpx;
  197. }
  198. .city-list {
  199. display: flex;
  200. justify-content: space-around;
  201. flex-wrap: wrap;
  202. .city {
  203. width: 156rpx;
  204. height: 68rpx;
  205. text-align: center;
  206. line-height: 68rpx;
  207. background-color: #f4f4f4;
  208. border-radius: 36rpx;
  209. margin-bottom: 20rpx;
  210. color: #222;
  211. font-size: 28rpx;
  212. }
  213. }
  214. .confirm {
  215. background-color: #f83224;
  216. border-radius: 40rpx;
  217. color: #fff;
  218. width: 702rpx;
  219. margin: 0 auto;
  220. }
  221. .picker-view {
  222. width: 100%;
  223. height: 180px;
  224. margin-top: 10px;
  225. }
  226. .item {
  227. height: 50px;
  228. }
  229. .text {
  230. line-height: 50px;
  231. text-align: center;
  232. }
  233. .picker-view-column {
  234. text-align: center;
  235. }
  236. </style>