初学Vue2,这个功能整整花了2天时间才搞定,github翻了一堆demo都没找到想要的,最终还是自己写了…

1、新建一个region.vue子组件,写入以下代码

  1. <template>
  2. <div id="app">
  3. <div class="mask" @click="selectaddress"></div>
  4. <div class="clearfix linkage">
  5. <div class="picker-toolbar">
  6. <span class="mint-datetime-action mint-datetime-cancel" @click="cancleaddress">取消</span>
  7. <span class="mint-datetime-action mint-datetime-cancel mini-title">选择投保城市</span>
  8. <span class="mint-datetime-action mint-datetime-confirm" @click="selectaddress">确定</span>
  9. </div>
  10. <mt-picker :slots="slots" value-key="name" @change="onValuesChange"></mt-picker>
  11. </div>
  12. </div>
  13. </div>
  14. </template>
  15. <script>
  16. import {address} from '../../api/cityData' //引入cityData数据
  17. const province = '广东';
  18. const city = '潮州';
  19. const area = '潮安区';
  20. export default {
  21. computed: {
  22. result() {
  23. return {
  24. province : this.yes_province.name, // 省
  25. city : this.yes_city.name, // 市
  26. area : this.yes_area.name, // 区
  27. code : this.yes_area.code, // 最后一个选择器的城市编码
  28. id : this.yes_area.id, // 最后一个选择器的城市ID
  29. province_car_prefix: this.yes_province.car_prefix, // 省级车牌前缀
  30. city_car_prefix : this.yes_city.car_prefix.substr(1), // 市级车牌前缀
  31. showLinkage : false // 关闭选择器
  32. }
  33. }
  34. },
  35. data() {
  36. return {
  37. yes_province:{name:'',id:'',id:'',car_prefix:''},
  38. yes_city:{name:'',id:'',id:'',car_prefix:''},
  39. yes_area:{name:'',id:'',id:'',car_prefix:''},
  40. province:{name:'',id:'',id:'',car_prefix:''},
  41. city:{name:'',id:'',id:'',car_prefix:''},
  42. area:{name:'',id:'',id:'',car_prefix:''},
  43. myAdress:null,
  44. flag : false, // 用于初始化参数
  45. slots: [
  46. {
  47. flex: 1,
  48. values: address,
  49. defaultIndex:0,
  50. className: 'slot1',
  51. textAlign: 'center'
  52. }, {
  53. divider: true,
  54. content: '-',
  55. className: 'slot2'
  56. }, {
  57. flex: 1,
  58. values: address[0].childs,
  59. defaultIndex:0,
  60. className: 'slot3',
  61. textAlign: 'center'
  62. }, {
  63. divider: true,
  64. content: '-',
  65. className: 'slot4'
  66. }, {
  67. flex: 1,
  68. values: address[0].childs[0].childs,
  69. defaultIndex:0,
  70. className: 'slot5',
  71. textAlign: 'center'
  72. }
  73. ]
  74. };
  75. },
  76. methods:{
  77. // 隐藏选择器
  78. cancleaddress: function () {
  79. this.$emit('getLinkage', this.result);
  80. },
  81. // 确认选择
  82. selectaddress: function () {
  83. // 触发父组件的getLinkage事件接收结果数据
  84. this.yes_province = this.province;
  85. this.yes_city = this.city;
  86. this.yes_area = this.area;
  87. this.$emit('getLinkage', this.result);
  88. },
  89. // 初始化数据
  90. runData: function () {
  91. // 赋默认值
  92. let city_array = '';
  93. let area_array = '';
  94. // 处理省级
  95. address.forEach((item, index)=>{
  96. if(item.name === province){
  97. console.log(item.name);
  98. // 选择省级默认值
  99. this.slots[0].defaultIndex = index;
  100. // 设置市级数据
  101. this.slots[2].values = address[index].childs;
  102. city_array = address[index].childs;
  103. this.province = item;
  104. }
  105. });
  106. // 处理市级
  107. city_array.forEach((item, index)=>{
  108. if(item.name === city){
  109. console.log(item.name);
  110. // 选择市级默认值
  111. this.slots[2].defaultIndex = index;
  112. // 设置区级数据
  113. this.slots[4].values = city_array[index].childs;
  114. area_array = city_array[index].childs;
  115. this.city = item;
  116. }
  117. });
  118. // 处理区级
  119. area_array.forEach((item, index)=>{
  120. if(item.name === area){
  121. console.log(item.name);
  122. // 选择区级默认值
  123. this.slots[4].defaultIndex = index;
  124. this.area = item;
  125. }
  126. });
  127. // 第一次要返回默认值
  128. this.selectaddress();
  129. },
  130. onValuesChange(picker, values) {
  131. picker.setSlotValues(1, values[0].childs);
  132. let town = [];
  133. if(values[1]){
  134. town = values[1].childs;
  135. }
  136. picker.setSlotValues(2,town);
  137. if(this.flag == false){
  138. this.flag = true;
  139. this.runData();
  140. } else {
  141. // 设置被选中的值
  142. let array = picker.getValues();
  143. // 循环中处理选中参数
  144. array.forEach((item, index)=>{
  145. // 省
  146. if (index == 0) {
  147. this.province = item;
  148. // 市
  149. } else if (index == 1) {
  150. this.city = item;
  151. // 区
  152. } else if (index == 2) {
  153. this.area = item;
  154. }
  155. });
  156. }
  157. }
  158. }
  159. }
  160. </script>
  161. <style scoped>
  162. .mint-datetime-action{width:32%!important;}
  163. .mini-title{color:#333}
  164. .mask{position:fixed;top:0;left:0;height:100%;width:100%;background:rgba(0,0,0,0.5)}
  165. .linkage{position:fixed;bottom:0;width:100%;background:white;left:0;overflow:hidden}
  166. .clearfix:after{content:".";display:block;height:0;clear:both;visibility:hidden}
  167. .pickerWrapper{width:33.3%;float:left}
  168. </style>

2、新建一个Test.vue父组件,写入以下代码:

  1. <template>
  2. <div id="app">
  3. <transition name="fade">
  4. <region @getLinkage='_showChildMsg' v-show="showLinkage"></region>
  5. </transition>
  6. <p v-html="region_name" class="getCity" @click="_showLinkage"></p>
  7. <p v-html="region_id" class="getCity" ></p>
  8. <p v-html="region_code" class="getCity" ></p>
  9. <p v-html="region_car_prefix" class="getCity" ></p>
  10. </div>
  11. </template>
  12. <script type="text/ecmascript-6">
  13. import Region from './region' // 引入地区组件
  14. export default {
  15. data(){
  16. return{
  17. // 默认情况下地区子组件是隐藏的
  18. showLinkage:false,
  19. // 地区名
  20. region_name:'',
  21. // 地区ID
  22. region_id:'',
  23. // 地区代码
  24. region_code:'',
  25. // 车牌前缀
  26. region_car_prefix:''
  27. }
  28. },
  29. // 注册子组件
  30. components:{
  31. 'region':Region,
  32. },
  33. methods:{
  34. // 接收地区子组件数据
  35. _showChildMsg(msg){
  36. this.region_name = msg.province+msg.city+msg.area
  37. this.region_id = msg.id
  38. this.region_code = msg.code
  39. this.region_car_prefix = msg.province_car_prefix + '-' + msg.city_car_prefix
  40. this.showLinkage = false
  41. },
  42. // 显示地区子组件
  43. _showLinkage(){
  44. this.showLinkage = true
  45. }
  46. }
  47. }
  48. </script>
  49. <style scoped>
  50. .getCity{font-size: 20px;}
  51. .fade-enter-active, .fade-leave-active {transition: opacity .5s;}
  52. .fade-enter, .fade-leave-to {opacity: 0;}
  53. </style>

3、最后是上传我们需要使用到的地区数据文件:

位置在:/项目目录/api/cityData.js
具体的数据包可以直接下载,数据库由于项目原因就不分享了。
下载地址:https://blog.junphp.com/github/cityData.js

4、功能介绍

该demo实现了:
1、地区选择器可以设置默认值
2、子组件初始化时,可以将默认值对应的参数返回给父组件
3、地区选择器带有确认取消按钮。

效果如下图: