picker-city.vue 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. <template>
  2. <view>
  3. <view class="appAuth" @click="onShowProvince"></view>
  4. <block v-if="enable">
  5. <view class="picker-view" :class="{pickerViewA:anims}">
  6. <view class="picker-view__pane">
  7. <text @click="cityCancel">取消</text>
  8. <text @click="citySure">确认</text>
  9. </view>
  10. <picker-view class="pick-view__group" @change="cityChange" @pickstart="chooseStart" @pickend="chooseEnd" :value="pickerValue">
  11. <picker-view-column indicator-class="item_active">
  12. <view v-for="item in provinces" class="picker-item" :key="item.id" :data-id="item.id">{{item.name}}</view>
  13. </picker-view-column>
  14. <picker-view-column>
  15. <view v-for="item in citys" class="picker-item" :key="item.id" :data-id="item.id">{{item.name}}</view>
  16. </picker-view-column>
  17. <picker-view-column>
  18. <view v-for="item in areas" class="picker-item" :key="item.id" :data-id="item.id">{{item.name}}</view>
  19. </picker-view-column>
  20. </picker-view>
  21. </view>
  22. </block>
  23. </view>
  24. </template>
  25. <script>
  26. import $api from '@/static/js/api.js'
  27. export default {
  28. name: "picker-city",
  29. props: {
  30. p_c_a: {
  31. type: [Object,String,Boolean],
  32. default: {},
  33. },//省、市、区 用于记忆查询
  34. },
  35. data() {
  36. return {
  37. id: 0,
  38. enable:false,
  39. anims:false,
  40. pickerValue: [0, 0, 0], // 地址选择器省市区 暂存 currentIndex
  41. provinces:[], // 省 一级地址
  42. citys: [], // 市 二级地址
  43. areas: [], // 区 三级地址
  44. isCanConfirm: true //是否禁止在第一列滚动期间点击确定提交数据
  45. };
  46. },
  47. created() {
  48. let that=this;
  49. that.getcity()
  50. },
  51. methods:{
  52. getcity() {
  53. var that = this
  54. $api.req({
  55. url: 'area',
  56. method: 'GET',
  57. data: {
  58. parent_id: that.id
  59. }
  60. }, function(res) {
  61. that.provinces = res.data
  62. })
  63. },
  64. // 显示选择器
  65. async onShowProvince(){
  66. console.log('显示地址选择器');
  67. let p_c_a=this.p_c_a;
  68. let that=this;
  69. /* if(p_c_a && (p_c_a.p_name || p_c_a.c_name || p_c_a.a_name)){
  70. that.provincePicker(p_c_a);
  71. }else{
  72. that.initPick();
  73. } */
  74. that.initPick();
  75. that.enable=true;
  76. that.$nextTick(()=>{
  77. that.anims=true;
  78. })
  79. },
  80. // 隐藏选择器
  81. async onHideProvince(){
  82. let that=this
  83. that.anims=false
  84. setTimeout(function() {
  85. that.pickerValue=[0,0,0]
  86. that.enable=false
  87. }, 350);
  88. },
  89. // 初始选择器
  90. initPick(){
  91. // 默认联动显示北京
  92. var that = this
  93. var id = that.provinces[0].id
  94. $api.req({
  95. url: 'area',
  96. method: 'GET',
  97. data: {
  98. parent_id: id
  99. }
  100. }, function(res2) {
  101. that.citys = res2.data
  102. $api.req({
  103. url: 'area',
  104. method: 'GET',
  105. data: {
  106. parent_id: that.citys[0].id
  107. }
  108. }, function(res3) {
  109. that.areas = res3.data
  110. })
  111. })
  112. },
  113. //
  114. chooseStart(e) {
  115. this.isCanConfirm=false
  116. },
  117. chooseEnd(e) {
  118. this.isCanConfirm=true
  119. },
  120. cityChange(e) {
  121. let arr = e.detail.value;
  122. // console.log("arr: " + JSON.stringify(arr));
  123. let provinceNum = arr[0];
  124. let cityNum = arr[1];
  125. let areaNum = arr[2];
  126. var p_id = provinces[provinceNum].id;
  127. if (this.pickerValue[0] !== provinceNum) {
  128. this.pickerValue=[provinceNum, 0, 0];
  129. this.citys=citys[p_id];
  130. this.areas=areas[citys[p_id][0].id]
  131. } else if (this.pickerValue[1] !== cityNum) {
  132. var c_id = citys[p_id][cityNum].id;
  133. this.pickerValue=[provinceNum, cityNum, 0];
  134. this.areas=areas[c_id];
  135. } else {
  136. this.pickerValue=[provinceNum, cityNum, areaNum];
  137. }
  138. },
  139. // 点击地区选择取消按钮
  140. cityCancel(e) {
  141. this.onHideProvince()
  142. },
  143. // 点击地区选择确定按钮
  144. citySure(e) {
  145. if (this.isCanConfirm) {
  146. var arr = this.pickerValue
  147. let obj={
  148. provinces: this.provinces[arr[0]] || null,
  149. city: this.citys[arr[1]] || null,
  150. area: this.areas[arr[2]] || null,
  151. }
  152. console.log("省市区: " + JSON.stringify(obj));
  153. this.$emit('confirm', obj)
  154. this.onHideProvince()
  155. }
  156. },
  157. // 记忆查询 地址逆解析
  158. async provincePicker({p_name='',c_name='',a_name=''}){
  159. let that=this
  160. var arr_picker=[0,0,0]
  161. var p_id=provinces.find((item,index)=>item.name==p_name || item.fullname==p_name).id
  162. arr_picker[0]=provinces.findIndex(item=>item.name==p_name || item.fullname==p_name)
  163. var c_id=citys[p_id].find((item,index)=>item.name==c_name || item.fullname==c_name).id
  164. arr_picker[1]=citys[p_id].findIndex(item=>item.name==c_name || item.fullname==c_name)
  165. var a_id=areas[c_id].find((item,index)=>item.name==a_name || item.fullname==a_name).id
  166. arr_picker[2]=areas[c_id].findIndex(item=>item.name==a_name || item.fullname==a_name)
  167. this.provinces=provinces;
  168. this.citys=citys[p_id];
  169. this.areas=areas[c_id];
  170. that.$nextTick(()=>{
  171. that.pickerValue=arr_picker
  172. })
  173. },
  174. }
  175. }
  176. </script>
  177. <style scoped>
  178. .appAuth{
  179. position: absolute;
  180. width: 100%;
  181. height: 100%;
  182. opacity: 0;
  183. top: 0;
  184. left: 0;
  185. right: 0;
  186. bottom: 0;
  187. z-index: 100;
  188. }
  189. /* 地区级联选择器 */
  190. .picker-view {
  191. position: fixed;
  192. width: 100%;
  193. display: flex;
  194. background-color: #ffffff;
  195. flex-direction: column;
  196. justify-content: center;
  197. align-items: center;
  198. left: 0rpx;
  199. bottom: -100%;
  200. transition: all 0.35s linear;
  201. border-top: 1rpx solid #eeeeee;
  202. }
  203. .pickerViewA{
  204. bottom: 0rpx;
  205. }
  206. .picker-item {
  207. line-height: 70rpx;
  208. margin-left: 5rpx;
  209. margin-right: 5rpx;
  210. text-align: center;
  211. }
  212. .picker-view__pane {
  213. height: 100rpx;
  214. width: 100%;
  215. padding: 20rpx 32rpx;
  216. display: flex;
  217. justify-content: space-between;
  218. align-items: center;
  219. box-sizing: border-box;
  220. }
  221. .picker-view__pane text {
  222. color: #00cc88;
  223. font-size: 30rpx;
  224. }
  225. .pick-view__group {
  226. width: 96%;
  227. height: 450rpx;
  228. }
  229. </style>