uview-pro 0.2.1 → 0.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (243) hide show
  1. package/changelog.md +501 -481
  2. package/components/common/props.ts +22 -22
  3. package/components/u-action-sheet/types.ts +35 -35
  4. package/components/u-action-sheet/u-action-sheet.vue +160 -160
  5. package/components/u-alert-tips/types.ts +39 -39
  6. package/components/u-alert-tips/u-alert-tips.vue +212 -212
  7. package/components/u-avatar/types.ts +34 -34
  8. package/components/u-avatar/u-avatar.vue +193 -193
  9. package/components/u-avatar-cropper/types.ts +23 -23
  10. package/components/u-avatar-cropper/u-avatar-cropper.vue +286 -286
  11. package/components/u-avatar-cropper/weCropper.d.ts +62 -62
  12. package/components/u-avatar-cropper/weCropper.js +1253 -1253
  13. package/components/u-avatar-cropper/weCropper.ts +1255 -1255
  14. package/components/u-back-top/types.ts +39 -39
  15. package/components/u-back-top/u-back-top.vue +125 -125
  16. package/components/u-badge/types.ts +36 -36
  17. package/components/u-badge/u-badge.vue +165 -165
  18. package/components/u-button/types.ts +66 -66
  19. package/components/u-button/u-button.vue +556 -556
  20. package/components/u-calendar/types.ts +73 -73
  21. package/components/u-calendar/u-calendar.vue +638 -638
  22. package/components/u-car-keyboard/types.ts +12 -12
  23. package/components/u-car-keyboard/u-car-keyboard.vue +234 -234
  24. package/components/u-card/types.ts +59 -59
  25. package/components/u-card/u-card.vue +194 -194
  26. package/components/u-cell-group/types.ts +17 -17
  27. package/components/u-cell-group/u-cell-group.vue +50 -50
  28. package/components/u-cell-item/types.ts +54 -54
  29. package/components/u-cell-item/u-cell-item.vue +202 -202
  30. package/components/u-checkbox/types.ts +31 -31
  31. package/components/u-checkbox/u-checkbox.vue +267 -267
  32. package/components/u-checkbox-group/types.ts +32 -32
  33. package/components/u-checkbox-group/u-checkbox-group.vue +79 -79
  34. package/components/u-circle-progress/types.ts +52 -52
  35. package/components/u-circle-progress/u-circle-progress.vue +187 -187
  36. package/components/u-city-select/types.ts +20 -20
  37. package/components/u-city-select/u-city-select.vue +236 -236
  38. package/components/u-col/types.ts +30 -30
  39. package/components/u-col/u-col.vue +123 -123
  40. package/components/u-collapse/types.ts +33 -33
  41. package/components/u-collapse/u-collapse.vue +69 -69
  42. package/components/u-collapse-item/types.ts +27 -27
  43. package/components/u-collapse-item/u-collapse-item.vue +213 -201
  44. package/components/u-column-notice/types.ts +48 -48
  45. package/components/u-column-notice/u-column-notice.vue +176 -176
  46. package/components/u-count-down/types.ts +42 -42
  47. package/components/u-count-down/u-count-down.vue +258 -258
  48. package/components/u-count-to/types.ts +32 -32
  49. package/components/u-count-to/u-count-to.vue +241 -241
  50. package/components/u-divider/types.ts +31 -31
  51. package/components/u-divider/u-divider.vue +121 -121
  52. package/components/u-dropdown/types.ts +32 -32
  53. package/components/u-dropdown/u-dropdown.vue +289 -289
  54. package/components/u-dropdown-item/types.ts +27 -27
  55. package/components/u-dropdown-item/u-dropdown-item.vue +123 -123
  56. package/components/u-empty/types.ts +36 -36
  57. package/components/u-empty/u-empty.vue +88 -88
  58. package/components/u-field/types.ts +69 -69
  59. package/components/u-field/u-field.vue +354 -354
  60. package/components/u-form/u-form.vue +132 -132
  61. package/components/u-form-item/u-form-item.vue +417 -417
  62. package/components/u-full-screen/types.ts +14 -14
  63. package/components/u-full-screen/u-full-screen.vue +82 -82
  64. package/components/u-gap/types.ts +18 -18
  65. package/components/u-gap/u-gap.vue +40 -40
  66. package/components/u-grid/types.ts +19 -19
  67. package/components/u-grid/u-grid.vue +93 -93
  68. package/components/u-grid-item/types.ts +16 -16
  69. package/components/u-grid-item/u-grid-item.vue +130 -130
  70. package/components/u-icon/types.ts +62 -62
  71. package/components/u-icon/u-icon.vue +281 -281
  72. package/components/u-image/types.ts +51 -51
  73. package/components/u-image/u-image.vue +222 -222
  74. package/components/u-index-anchor/types.ts +16 -16
  75. package/components/u-index-anchor/u-index-anchor.vue +86 -86
  76. package/components/u-index-list/types.ts +43 -43
  77. package/components/u-index-list/u-index-list.vue +355 -355
  78. package/components/u-input/types.ts +140 -140
  79. package/components/u-input/u-input.vue +264 -264
  80. package/components/u-keyboard/types.ts +40 -40
  81. package/components/u-keyboard/u-keyboard.vue +158 -158
  82. package/components/u-lazy-load/types.ts +37 -37
  83. package/components/u-lazy-load/u-lazy-load.vue +233 -233
  84. package/components/u-line/types.ts +44 -44
  85. package/components/u-line/u-line.vue +59 -59
  86. package/components/u-line-progress/types.ts +58 -58
  87. package/components/u-line-progress/u-line-progress.vue +109 -109
  88. package/components/u-link/types.ts +43 -43
  89. package/components/u-link/u-link.vue +75 -75
  90. package/components/u-loading/types.ts +35 -35
  91. package/components/u-loading/u-loading.vue +90 -90
  92. package/components/u-loading-popup/types.ts +26 -26
  93. package/components/u-loading-popup/u-loading-popup.vue +239 -239
  94. package/components/u-loadmore/types.ts +79 -79
  95. package/components/u-loadmore/u-loadmore.vue +140 -140
  96. package/components/u-mask/types.ts +43 -43
  97. package/components/u-mask/u-mask.vue +106 -106
  98. package/components/u-message-input/types.ts +74 -74
  99. package/components/u-message-input/u-message-input.vue +255 -255
  100. package/components/u-modal/types.ts +118 -118
  101. package/components/u-modal/u-modal.vue +204 -204
  102. package/components/u-navbar/types.ts +103 -103
  103. package/components/u-navbar/u-navbar.vue +226 -226
  104. package/components/u-no-network/image.ts +2 -2
  105. package/components/u-no-network/types.ts +28 -28
  106. package/components/u-no-network/u-no-network.vue +290 -290
  107. package/components/u-notice-bar/types.ts +111 -111
  108. package/components/u-notice-bar/u-notice-bar.vue +174 -174
  109. package/components/u-number-box/types.ts +42 -42
  110. package/components/u-number-box/u-number-box.vue +312 -312
  111. package/components/u-number-keyboard/types.ts +26 -26
  112. package/components/u-number-keyboard/u-number-keyboard.vue +166 -166
  113. package/components/u-picker/types.ts +123 -123
  114. package/components/u-picker/u-picker.vue +637 -637
  115. package/components/u-popup/types.ts +59 -59
  116. package/components/u-popup/u-popup.vue +359 -359
  117. package/components/u-radio/types.ts +25 -25
  118. package/components/u-radio/u-radio.vue +258 -258
  119. package/components/u-radio-group/types.ts +29 -29
  120. package/components/u-radio-group/u-radio-group.vue +98 -98
  121. package/components/u-rate/types.ts +40 -40
  122. package/components/u-rate/u-rate.vue +234 -234
  123. package/components/u-read-more/types.ts +35 -35
  124. package/components/u-read-more/u-read-more.vue +150 -150
  125. package/components/u-root-portal/u-root-portal.vue +54 -0
  126. package/components/u-row/types.ts +20 -20
  127. package/components/u-row/u-row.vue +87 -87
  128. package/components/u-row-notice/types.ts +39 -39
  129. package/components/u-row-notice/u-row-notice.vue +213 -213
  130. package/components/u-safe-bottom/u-safe-bottom.vue +46 -46
  131. package/components/u-search/types.ts +53 -53
  132. package/components/u-search/u-search.vue +256 -256
  133. package/components/u-section/types.ts +32 -32
  134. package/components/u-section/u-section.vue +125 -125
  135. package/components/u-select/types.ts +43 -43
  136. package/components/u-select/u-select.vue +361 -361
  137. package/components/u-skeleton/types.ts +20 -20
  138. package/components/u-skeleton/u-skeleton.vue +205 -205
  139. package/components/u-slider/types.ts +32 -32
  140. package/components/u-slider/u-slider.vue +238 -238
  141. package/components/u-status-bar/u-status-bar.vue +65 -65
  142. package/components/u-steps/types.ts +28 -28
  143. package/components/u-steps/u-steps.vue +160 -160
  144. package/components/u-sticky/types.ts +22 -22
  145. package/components/u-sticky/u-sticky.vue +159 -159
  146. package/components/u-subsection/types.ts +36 -36
  147. package/components/u-subsection/u-subsection.vue +328 -328
  148. package/components/u-swipe-action/types.ts +50 -50
  149. package/components/u-swipe-action/u-swipe-action.vue +253 -253
  150. package/components/u-swiper/types.ts +47 -47
  151. package/components/u-swiper/u-swiper.vue +266 -266
  152. package/components/u-switch/types.ts +28 -28
  153. package/components/u-switch/u-switch.vue +136 -136
  154. package/components/u-tabbar/types.ts +36 -36
  155. package/components/u-tabbar/u-tabbar.vue +280 -280
  156. package/components/u-table/types.ts +25 -25
  157. package/components/u-table/u-table.vue +55 -55
  158. package/components/u-tabs/types.ts +51 -51
  159. package/components/u-tabs/u-tabs.vue +284 -284
  160. package/components/u-tabs-swiper/types.ts +53 -53
  161. package/components/u-tabs-swiper/u-tabs-swiper.vue +379 -379
  162. package/components/u-tag/types.ts +37 -37
  163. package/components/u-tag/u-tag.vue +244 -244
  164. package/components/u-td/types.ts +12 -12
  165. package/components/u-td/u-td.vue +87 -87
  166. package/components/u-text/types.ts +69 -69
  167. package/components/u-text/u-text.vue +326 -326
  168. package/components/u-th/types.ts +12 -12
  169. package/components/u-th/u-th.vue +81 -81
  170. package/components/u-time-line/u-time-line.vue +39 -39
  171. package/components/u-time-line-item/types.ts +14 -14
  172. package/components/u-time-line-item/u-time-line-item.vue +78 -78
  173. package/components/u-toast/types.ts +36 -36
  174. package/components/u-toast/u-toast.vue +233 -233
  175. package/components/u-top-tips/types.ts +14 -14
  176. package/components/u-top-tips/u-top-tips.vue +113 -113
  177. package/components/u-tr/types.ts +8 -8
  178. package/components/u-tr/u-tr.vue +24 -24
  179. package/components/u-upload/types.ts +74 -74
  180. package/components/u-upload/u-upload.vue +545 -545
  181. package/components/u-verification-code/types.ts +22 -22
  182. package/components/u-verification-code/u-verification-code.vue +164 -164
  183. package/components/u-waterfall/types.ts +16 -16
  184. package/components/u-waterfall/u-waterfall.vue +175 -175
  185. package/iconfont.css +912 -912
  186. package/index.scss +25 -25
  187. package/index.ts +29 -29
  188. package/libs/config/config.ts +26 -26
  189. package/libs/config/zIndex.ts +37 -37
  190. package/libs/css/color.scss +155 -155
  191. package/libs/css/common.scss +178 -178
  192. package/libs/css/style.components.scss +16 -16
  193. package/libs/css/style.h5.scss +8 -8
  194. package/libs/css/style.mp.scss +72 -72
  195. package/libs/css/style.nvue.scss +15 -15
  196. package/libs/css/style.vue.scss +188 -188
  197. package/libs/function/$parent.ts +21 -21
  198. package/libs/function/addUnit.ts +13 -13
  199. package/libs/function/color.ts +37 -37
  200. package/libs/function/colorGradient.ts +125 -125
  201. package/libs/function/debounce.ts +28 -28
  202. package/libs/function/deepClone.ts +39 -39
  203. package/libs/function/deepMerge.ts +34 -34
  204. package/libs/function/getParent.ts +59 -59
  205. package/libs/function/getRect.ts +26 -26
  206. package/libs/function/guid.ts +42 -42
  207. package/libs/function/md5.ts +391 -391
  208. package/libs/function/parent.ts +21 -21
  209. package/libs/function/queryParams.ts +60 -60
  210. package/libs/function/random.ts +16 -16
  211. package/libs/function/randomArray.ts +11 -11
  212. package/libs/function/route.ts +118 -118
  213. package/libs/function/styleUtils.ts +83 -83
  214. package/libs/function/sys.ts +15 -15
  215. package/libs/function/test.ts +285 -285
  216. package/libs/function/throttle.ts +31 -31
  217. package/libs/function/timeFormat.ts +54 -54
  218. package/libs/function/timeFrom.ts +48 -48
  219. package/libs/function/toast.ts +14 -14
  220. package/libs/function/trim.ts +21 -21
  221. package/libs/function/type2icon.ts +36 -36
  222. package/libs/hooks/index.ts +3 -3
  223. package/libs/hooks/useEmitter.ts +77 -77
  224. package/libs/hooks/useParent.ts +31 -31
  225. package/libs/hooks/useRect.ts +33 -33
  226. package/libs/index.ts +291 -291
  227. package/libs/request/auto-http.ts +76 -76
  228. package/libs/request/index.ts +223 -223
  229. package/libs/store/index.ts +88 -88
  230. package/libs/util/async-validator.d.ts +62 -62
  231. package/libs/util/async-validator.js +1 -1
  232. package/libs/util/calendar.d.ts +57 -57
  233. package/libs/util/emitter.ts +102 -102
  234. package/libs/util/mitt.ts +115 -115
  235. package/libs/util/parent.ts +20 -20
  236. package/package.json +1 -1
  237. package/readme.md +239 -237
  238. package/theme.scss +38 -38
  239. package/types/components.d.ts +97 -96
  240. package/types/global.d.ts +295 -295
  241. package/types/ignore-errors.d.ts +30 -30
  242. package/types/index.d.ts +19 -19
  243. package/types/uni-app.d.ts +63 -63
@@ -1,637 +1,637 @@
1
- <template>
2
- <u-popup
3
- :maskCloseAble="maskCloseAble"
4
- mode="bottom"
5
- :popup="false"
6
- v-model="popupValue"
7
- length="auto"
8
- :safeAreaInsetBottom="safeAreaInsetBottom"
9
- @close="close"
10
- :z-index="uZIndex"
11
- >
12
- <view class="u-datetime-picker">
13
- <view class="u-picker-header" @touchmove.stop.prevent="">
14
- <view class="u-btn-picker u-btn-picker--tips" :style="{ color: cancelColor }" hover-class="u-opacity" :hover-stay-time="150" @tap="getResult('cancel')">{{
15
- cancelText
16
- }}</view>
17
- <view class="u-picker__title">{{ title }}</view>
18
- <view
19
- class="u-btn-picker u-btn-picker--primary"
20
- :style="{ color: moving ? cancelColor : confirmColor }"
21
- hover-class="u-opacity"
22
- :hover-stay-time="150"
23
- @touchmove.stop=""
24
- @tap.stop="getResult('confirm')"
25
- >
26
- {{ confirmText }}
27
- </view>
28
- </view>
29
- <view class="u-picker-body">
30
- <picker-view v-if="mode == 'region'" :value="valueArr" @change="change" class="u-picker-view" @pickstart="pickstart" @pickend="pickend">
31
- <picker-view-column v-if="!reset && params.province">
32
- <view class="u-column-item" v-for="(item, index) in provinces" :key="index">
33
- <view class="u-line-1">{{ item.label }}</view>
34
- </view>
35
- </picker-view-column>
36
- <picker-view-column v-if="!reset && params.city">
37
- <view class="u-column-item" v-for="(item, index) in citysRef" :key="index">
38
- <view class="u-line-1">{{ item.label }}</view>
39
- </view>
40
- </picker-view-column>
41
- <picker-view-column v-if="!reset && params.area">
42
- <view class="u-column-item" v-for="(item, index) in areasRef" :key="index">
43
- <view class="u-line-1">{{ item.label }}</view>
44
- </view>
45
- </picker-view-column>
46
- </picker-view>
47
- <picker-view v-else-if="mode == 'time'" :value="valueArr" @change="change" class="u-picker-view" @pickstart="pickstart" @pickend="pickend">
48
- <picker-view-column v-if="!reset && params.year">
49
- <view class="u-column-item" v-for="(item, index) in years" :key="index">
50
- {{ item }}
51
- <text class="u-text" v-if="showTimeTag">年</text>
52
- </view>
53
- </picker-view-column>
54
- <picker-view-column v-if="!reset && params.month">
55
- <view class="u-column-item" v-for="(item, index) in months" :key="index">
56
- {{ formatNumber(item) }}
57
- <text class="u-text" v-if="showTimeTag">月</text>
58
- </view>
59
- </picker-view-column>
60
- <picker-view-column v-if="!reset && params.day">
61
- <view class="u-column-item" v-for="(item, index) in days" :key="index">
62
- {{ formatNumber(item) }}
63
- <text class="u-text" v-if="showTimeTag">日</text>
64
- </view>
65
- </picker-view-column>
66
- <picker-view-column v-if="!reset && params.hour">
67
- <view class="u-column-item" v-for="(item, index) in hours" :key="index">
68
- {{ formatNumber(item) }}
69
- <text class="u-text" v-if="showTimeTag">时</text>
70
- </view>
71
- </picker-view-column>
72
- <picker-view-column v-if="!reset && params.minute">
73
- <view class="u-column-item" v-for="(item, index) in minutes" :key="index">
74
- {{ formatNumber(item) }}
75
- <text class="u-text" v-if="showTimeTag">分</text>
76
- </view>
77
- </picker-view-column>
78
- <picker-view-column v-if="!reset && params.second">
79
- <view class="u-column-item" v-for="(item, index) in seconds" :key="index">
80
- {{ formatNumber(item) }}
81
- <text class="u-text" v-if="showTimeTag">秒</text>
82
- </view>
83
- </picker-view-column>
84
- </picker-view>
85
- <picker-view v-else-if="mode == 'selector'" :value="valueArr" @change="change" class="u-picker-view" @pickstart="pickstart" @pickend="pickend">
86
- <picker-view-column v-if="!reset">
87
- <view class="u-column-item" v-for="(item, index) in range" :key="index">
88
- <view class="u-line-1">{{ getItemValue(item, 'selector') }}</view>
89
- </view>
90
- </picker-view-column>
91
- </picker-view>
92
- <picker-view v-else-if="mode == 'multiSelector'" :value="valueArr" @change="change" class="u-picker-view" @pickstart="pickstart" @pickend="pickend">
93
- <picker-view-column v-if="!reset" v-for="(item, index) in range" :key="index">
94
- <view class="u-column-item" v-for="(item1, index1) in item" :key="index1">
95
- <view class="u-line-1">{{ getItemValue(item1, 'multiSelector') }}</view>
96
- </view>
97
- </picker-view-column>
98
- </picker-view>
99
- </view>
100
- </view>
101
- </u-popup>
102
- </template>
103
-
104
- <script setup lang="ts">
105
- import { ref, computed, watch, onMounted, nextTick } from 'vue';
106
- import provinces from '../../libs/util/province';
107
- import citys from '../../libs/util/city';
108
- import areas from '../../libs/util/area';
109
- import { PickerProps } from './types';
110
-
111
- defineOptions({
112
- name: 'u-picker'
113
- });
114
-
115
- /**
116
- * picker picker弹出选择器
117
- * @description 此选择器有两种弹出模式:一是时间模式,可以配置年,日,月,时,分,秒参数 二是地区模式,可以配置省,市,区参数
118
- * @tutorial https://uviewpro.cn/zh/components/picker.html
119
- * @property {Object} params 需要显示的参数,见官网说明
120
- * @property {String} mode 模式选择,region-地区类型,time-时间类型(默认time)
121
- * @property {String Number} start-year 可选的开始年份,mode=time时有效(默认1950)
122
- * @property {String Number} end-year 可选的结束年份,mode=time时有效(默认2050)
123
- * @property {Boolean} safe-area-inset-bottom 是否开启底部安全区适配(默认false)
124
- * @property {Boolean} show-time-tag 时间模式时,是否显示后面的年月日中文提示
125
- * @property {String} cancel-color 取消按钮的颜色(默认#606266)
126
- * @property {String} confirm-color 确认按钮的颜色(默认#2979ff)
127
- * @property {String} default-time 默认选中的时间,mode=time时有效
128
- * @property {String} confirm-text 确认按钮的文字
129
- * @property {String} cancel-text 取消按钮的文字
130
- * @property {String} default-region 默认选中的地区,中文形式,mode=region时有效
131
- * @property {String} default-code 默认选中的地区,编号形式,mode=region时有效
132
- * @property {Boolean} mask-close-able 是否允许通过点击遮罩关闭Picker(默认true)
133
- * @property {String Number} z-index 弹出时的z-index值(默认1075)
134
- * @property {Array} default-selector 数组形式,其中每一项表示选择了range对应项中的第几个
135
- * @property {Array} range 自定义选择的数据,mode=selector或mode=multiSelector时有效
136
- * @property {String} range-key 当range参数的元素为对象时,指定Object中的哪个key的值作为选择器显示内容
137
- * @event {Function} confirm 点击确定按钮,返回当前选择的值
138
- * @event {Function} cancel 点击取消按钮,返回当前选择的值
139
- * @example <u-picker v-model="show" mode="time"></u-picker>
140
- */
141
- const props = defineProps(PickerProps);
142
-
143
- const popupValue = computed({
144
- get: () => props.modelValue,
145
- set: (val: boolean) => emit('update:modelValue', val)
146
- });
147
-
148
- const emit = defineEmits(['update:modelValue', 'confirm', 'cancel', 'columnchange']);
149
-
150
- // 主要数据
151
- const years = ref<number[]>([]);
152
- const months = ref<number[]>([]);
153
- const days = ref<number[]>([]);
154
- const hours = ref<number[]>([]);
155
- const minutes = ref<number[]>([]);
156
- const seconds = ref<number[]>([]);
157
- const year = ref<number>(0);
158
- const month = ref<number>(0);
159
- const day = ref<number>(0);
160
- const hour = ref<number>(0);
161
- const minute = ref<number>(0);
162
- const second = ref<number>(0);
163
- const reset = ref(false);
164
- const startDate = ref('');
165
- const endDate = ref('');
166
- const valueArr = ref<number[]>([]);
167
- const provincesRef = ref<any[]>(provinces);
168
- const citysRef = ref<any[]>(citys[0]);
169
- const areasRef = ref<any[]>(areas[0][0]);
170
- const province = ref<number>(0);
171
- const city = ref<number>(0);
172
- const area = ref<number>(0);
173
- // 列是否还在滑动中,微信小程序如果在滑动中就点确定,结果可能不准确
174
- const moving = ref(false);
175
- const multiSelectorValue = ref<number[]>([]);
176
-
177
- // 计算属性
178
- // 引用这几个变量,是为了监听其变化
179
- const propsChange = computed(() => `${props.mode}-${props.defaultTime}-${props.startYear}-${props.endYear}-${props.defaultRegion}-${props.areaCode}`);
180
- // 引用这几个变量,是为了监听其变化
181
- const regionChange = computed(() => `${province.value}-${city.value}`);
182
-
183
- const yearAndMonth = computed(() => `${year.value}-${month.value}`);
184
- // 如果用户有传递z-index值,优先使用
185
- const uZIndex = computed(() => (props.zIndex ? props.zIndex : 1075));
186
-
187
- // 监听
188
- watch(propsChange, () => {
189
- reset.value = true;
190
- setTimeout(() => init(), 10);
191
- });
192
-
193
- // 如果地区发生变化,为了让picker联动起来,必须重置this.citys和this.areas
194
- watch(regionChange, () => {
195
- citysRef.value = citys[province.value];
196
- areasRef.value = areas[province.value][city.value];
197
- });
198
-
199
- // watch监听月份的变化,实时变更日的天数,因为不同月份,天数不一样
200
- // 一个月可能有30,31天,甚至闰年2月的29天,平年2月28天
201
- watch(yearAndMonth, () => {
202
- if (props.params.year) setDays();
203
- });
204
- watch(
205
- () => props.modelValue,
206
- n => {
207
- if (n) {
208
- reset.value = true;
209
- setTimeout(() => init(), 10);
210
- }
211
- }
212
- );
213
-
214
- // 方法区
215
- /**
216
- * 标识滑动开始,只有微信小程序才有这样的事件
217
- */
218
- function pickstart() {
219
- // #ifdef MP-WEIXIN
220
- moving.value = true;
221
- // #endif
222
- }
223
- /**
224
- * 标识滑动结束
225
- */
226
- function pickend() {
227
- // #ifdef MP-WEIXIN
228
- moving.value = false;
229
- // #endif
230
- }
231
- /**
232
- * 对单列和多列形式的判断是否有传入变量的情况
233
- * @param item 当前项
234
- * @param mode 模式
235
- */
236
- function getItemValue(item: any, mode: string) {
237
- // 目前(2020-05-25)uni-app对微信小程序编译有错误,导致v-if为false中的内容也执行,错误导致
238
- // 单列模式或者多列模式中的getItemValue同时被执行,故在这里再加一层判断
239
- if (props.mode == mode) {
240
- return typeof item == 'object' ? item[props.rangeKey] : item;
241
- }
242
- }
243
- /**
244
- * 小于10前面补0,用于月份,日期,时分秒等
245
- */
246
- function formatNumber(num: number | string) {
247
- return +num < 10 ? '0' + num : String(num);
248
- }
249
- /**
250
- * 生成递进的数组
251
- */
252
- function generateArray(start: number, end: number) {
253
- // 转为数值格式,否则用户给end-year等传递字符串值时,下面的end+1会导致字符串拼接,而不是相加
254
- start = Number(start);
255
- end = Number(end);
256
- end = end > start ? end : start;
257
- // 生成数组,获取其中的索引,并剪出来
258
- return [...Array(end + 1).keys()].slice(start);
259
- }
260
- /**
261
- * 获取数组中指定值的索引
262
- */
263
- function getIndex(arr: any[], val: any) {
264
- let index = arr.indexOf(val);
265
- // 如果index为-1(即找不到index值),~(-1)=-(-1)-1=0,导致条件不成立
266
- return ~index ? index : 0;
267
- }
268
- /**
269
- * 日期时间处理,初始化各时间字段
270
- */
271
- function initTimeValue() {
272
- // 格式化时间,在IE浏览器(uni不存在此情况),无法识别日期间的"-"间隔符号
273
- let fdate = props.defaultTime.replace(/\-/g, '/');
274
- fdate = fdate && fdate.indexOf('/') == -1 ? `2020/01/01 ${fdate}` : fdate;
275
- let time: Date;
276
- if (fdate) time = new Date(fdate);
277
- else time = new Date();
278
- // 获取年日月时分秒
279
- year.value = time.getFullYear();
280
- month.value = Number(time.getMonth()) + 1;
281
- day.value = time.getDate();
282
- hour.value = time.getHours();
283
- minute.value = time.getMinutes();
284
- second.value = time.getSeconds();
285
- }
286
- /**
287
- * 初始化picker各列数据
288
- */
289
- function init() {
290
- valueArr.value = [];
291
- reset.value = false;
292
- if (props.mode == 'time') {
293
- initTimeValue();
294
- if (props.params.year) {
295
- valueArr.value.push(0);
296
- setYears();
297
- }
298
- if (props.params.month) {
299
- valueArr.value.push(0);
300
- setMonths();
301
- }
302
- if (props.params.day) {
303
- valueArr.value.push(0);
304
- setDays();
305
- }
306
- if (props.params.hour) {
307
- valueArr.value.push(0);
308
- setHours();
309
- }
310
- if (props.params.minute) {
311
- valueArr.value.push(0);
312
- setMinutes();
313
- }
314
- if (props.params.second) {
315
- valueArr.value.push(0);
316
- setSeconds();
317
- }
318
- } else if (props.mode == 'region') {
319
- if (props.params.province) {
320
- valueArr.value.push(0);
321
- setProvinces();
322
- }
323
- if (props.params.city) {
324
- valueArr.value.push(0);
325
- setCitys();
326
- }
327
- if (props.params.area) {
328
- valueArr.value.push(0);
329
- setAreas();
330
- }
331
- } else if (props.mode == 'selector') {
332
- valueArr.value = props.defaultSelector as number[];
333
- } else if (props.mode == 'multiSelector') {
334
- valueArr.value = props.defaultSelector as number[];
335
- multiSelectorValue.value = props.defaultSelector as number[];
336
- }
337
- nextTick(() => {});
338
- }
339
- /**
340
- * 设置年份列
341
- */
342
- function setYears() {
343
- // 获取年份集合
344
- years.value = generateArray(props.startYear as number, props.endYear as number);
345
- // 设置this.valueArr某一项的值,是为了让picker预选中某一个值
346
- valueArr.value.splice(valueArr.value.length - 1, 1, getIndex(years.value, year.value));
347
- }
348
- /**
349
- * 设置月份列
350
- */
351
- function setMonths() {
352
- months.value = generateArray(1, 12);
353
- valueArr.value.splice(valueArr.value.length - 1, 1, getIndex(months.value, month.value));
354
- }
355
- /**
356
- * 设置天数列
357
- */
358
- function setDays() {
359
- let totalDays = new Date(year.value, month.value, 0).getDate();
360
- days.value = generateArray(1, totalDays);
361
- let index = 0;
362
- // 这里不能使用类似setMonths()中的this.valueArr.splice(this.valueArr.length - 1, 1, xxx)做法
363
- // 因为this.month和this.year变化时,会触发watch中的this.setDays(),导致this.valueArr.length计算有误
364
- if (props.params.year && props.params.month) index = 2;
365
- else if (props.params.month) index = 1;
366
- else if (props.params.year) index = 1;
367
- else index = 0;
368
- // 当月份变化时,会导致日期的天数也会变化,如果原来选的天数大于变化后的天数,则重置为变化后的最大值
369
- // 比如原来选中3月31日,调整为2月后,日期变为最大29,这时如果day值继续为31显然不合理,于是将其置为29(picker-column从1开始)
370
- if (day.value > days.value.length) day.value = days.value.length;
371
- valueArr.value.splice(index, 1, getIndex(days.value, day.value));
372
- }
373
- /**
374
- * 设置小时列
375
- */
376
- function setHours() {
377
- hours.value = generateArray(0, 23);
378
- valueArr.value.splice(valueArr.value.length - 1, 1, getIndex(hours.value, hour.value));
379
- }
380
- /**
381
- * 设置分钟列
382
- */
383
- function setMinutes() {
384
- minutes.value = generateArray(0, 59);
385
- valueArr.value.splice(valueArr.value.length - 1, 1, getIndex(minutes.value, minute.value));
386
- }
387
- /**
388
- * 设置秒数列
389
- */
390
- function setSeconds() {
391
- seconds.value = generateArray(0, 59);
392
- valueArr.value.splice(valueArr.value.length - 1, 1, getIndex(seconds.value, second.value));
393
- }
394
- /**
395
- * 设置省份列
396
- */
397
- function setProvinces() {
398
- // 判断是否需要province参数
399
- if (!props.params.province) return;
400
- let tmp: any = '';
401
- let useCode = false;
402
- // 如果同时配置了defaultRegion和areaCode,优先使用areaCode参数
403
- if (props.areaCode.length) {
404
- tmp = props.areaCode[0];
405
- useCode = true;
406
- } else if (props.defaultRegion.length) tmp = props.defaultRegion[0];
407
- else tmp = 0;
408
- // 历遍省份数组匹配
409
- provinces.map((v: any, k: number) => {
410
- if (useCode ? v.value == tmp : v.label == tmp) {
411
- tmp = k;
412
- }
413
- });
414
- province.value = tmp;
415
- provincesRef.value = provinces;
416
- // 设置默认省份的值
417
- valueArr.value.splice(0, 1, province.value);
418
- }
419
- /**
420
- * 设置城市列
421
- */
422
- function setCitys() {
423
- if (!props.params.city) return;
424
- let tmp: any = '';
425
- let useCode = false;
426
- if (props.areaCode.length) {
427
- tmp = props.areaCode[1];
428
- useCode = true;
429
- } else if (props.defaultRegion.length) tmp = props.defaultRegion[1];
430
- else tmp = 0;
431
- // 历遍城市数组匹配
432
- citys[province.value].map((v: any, k: number) => {
433
- if (useCode ? v.value == tmp : v.label == tmp) {
434
- tmp = k;
435
- }
436
- });
437
- city.value = tmp;
438
- citysRef.value = citys[province.value];
439
- // 设置默认城市的值
440
- valueArr.value.splice(1, 1, city.value);
441
- }
442
- /**
443
- * 设置区县列
444
- */
445
- function setAreas() {
446
- if (!props.params.area) return;
447
- let tmp: any = '';
448
- let useCode = false;
449
- if (props.areaCode.length) {
450
- tmp = props.areaCode[2];
451
- useCode = true;
452
- } else if (props.defaultRegion.length) tmp = props.defaultRegion[2];
453
- else tmp = 0;
454
- // 历遍区县数组匹配
455
- areas[province.value][city.value].map((v: any, k: number) => {
456
- if (useCode ? v.value == tmp : v.label == tmp) {
457
- tmp = k;
458
- }
459
- });
460
- area.value = tmp;
461
- areasRef.value = areas[province.value][city.value];
462
- // 设置默认区县的值
463
- valueArr.value.splice(2, 1, area.value);
464
- }
465
- /**
466
- * 关闭picker弹窗
467
- */
468
- function close() {
469
- emit('update:modelValue', false);
470
- }
471
- /**
472
- * 用户更改picker的列选项
473
- * @param e 事件对象
474
- */
475
- function change(e: any) {
476
- valueArr.value = e.detail.value;
477
- let i = 0;
478
- if (props.mode == 'time') {
479
- // 这里使用i++,是因为valueArr数组的长度是不确定长度的,它根据params的值来配置长度
480
- // 进入if规则,i会加1,保证了能获取准确的值
481
- if (props.params.year) year.value = years.value[valueArr.value[i++]];
482
- if (props.params.month) month.value = months.value[valueArr.value[i++]];
483
- if (props.params.day) day.value = days.value[valueArr.value[i++]];
484
- if (props.params.hour) hour.value = hours.value[valueArr.value[i++]];
485
- if (props.params.minute) minute.value = minutes.value[valueArr.value[i++]];
486
- if (props.params.second) second.value = seconds.value[valueArr.value[i++]];
487
- } else if (props.mode == 'region') {
488
- if (props.params.province) province.value = valueArr.value[i++];
489
- if (props.params.city) city.value = valueArr.value[i++];
490
- if (props.params.area) area.value = valueArr.value[i++];
491
- } else if (props.mode == 'multiSelector') {
492
- let index: number | null = null;
493
- // 对比前后两个数组,寻找变更的是哪一列,如果某一个元素不同,即可判定该列发生了变化
494
- (props.defaultSelector as number[]).map((val: number, idx: number) => {
495
- if (val != e.detail.value[idx]) index = idx;
496
- });
497
- // 为了让用户对多列变化时,对动态设置其他列的变更
498
- if (index != null) {
499
- emit('columnchange', {
500
- column: index,
501
- index: e.detail.value[index]
502
- });
503
- }
504
- }
505
- }
506
- /**
507
- * 用户点击确定/取消按钮,获取结果
508
- * @param event 事件名
509
- */
510
- function getResult(event: string | null = null) {
511
- // #ifdef MP-WEIXIN
512
- // 微信小程序滑动中不允许点击确定,防止取值不准确
513
- if (moving.value) return;
514
- // #endif
515
- let result: any = {};
516
- // 只返回用户在params中配置了为true的字段
517
- if (props.mode == 'time') {
518
- if (props.params.year) result.year = formatNumber(year.value || 0); // 年
519
- if (props.params.month) result.month = formatNumber(month.value || 0); // 月
520
- if (props.params.day) result.day = formatNumber(day.value || 0); // 日
521
- if (props.params.hour) result.hour = formatNumber(hour.value || 0); // 时
522
- if (props.params.minute) result.minute = formatNumber(minute.value || 0); // 分
523
- if (props.params.second) result.second = formatNumber(second.value || 0); // 秒
524
- if (props.params.timestamp) result.timestamp = getTimestamp(); // 时间戳
525
- } else if (props.mode == 'region') {
526
- // 地区模式,返回省市区对象
527
- if (props.params.province) result.province = provinces[province.value];
528
- if (props.params.city) result.city = citys[province.value][city.value];
529
- if (props.params.area) result.area = areas[province.value][city.value][area.value];
530
- } else if (props.mode == 'selector') {
531
- // 单列选择器,直接返回选中下标数组
532
- result = valueArr.value;
533
- } else if (props.mode == 'multiSelector') {
534
- // 多列选择器,直接返回选中下标数组
535
- result = valueArr.value;
536
- }
537
- // 只允许 emit 已声明的事件类型
538
- if (event && ['update:modelValue', 'confirm', 'cancel', 'columnchange'].includes(event)) emit(event as 'update:modelValue' | 'confirm' | 'cancel' | 'columnchange', result);
539
- close();
540
- }
541
- /**
542
- * 获取时间戳(秒)
543
- * @returns {number} 时间戳
544
- */
545
- function getTimestamp() {
546
- // yyyy-mm-dd为安卓写法,不支持iOS,需要使用"/"分隔,才能二者兼容
547
- let time = year.value + '/' + month.value + '/' + day.value + ' ' + hour.value + ':' + minute.value + ':' + second.value;
548
- return new Date(time).getTime() / 1000;
549
- }
550
-
551
- // 组件挂载时初始化
552
- onMounted(() => {
553
- init();
554
- });
555
- </script>
556
-
557
- <style lang="scss" scoped>
558
- @import '../../libs/css/style.components.scss';
559
-
560
- .u-datetime-picker {
561
- position: relative;
562
- z-index: 999;
563
- }
564
-
565
- .u-picker-view {
566
- height: 100%;
567
- box-sizing: border-box;
568
- }
569
-
570
- .u-picker-header {
571
- width: 100%;
572
- height: 90rpx;
573
- padding: 0 40rpx;
574
- @include vue-flex;
575
- justify-content: space-between;
576
- align-items: center;
577
- box-sizing: border-box;
578
- font-size: 30rpx;
579
- background: #fff;
580
- position: relative;
581
- }
582
-
583
- .u-picker-header::after {
584
- content: '';
585
- position: absolute;
586
- border-bottom: 1rpx solid #eaeef1;
587
- -webkit-transform: scaleY(0.5);
588
- transform: scaleY(0.5);
589
- bottom: 0;
590
- right: 0;
591
- left: 0;
592
- }
593
-
594
- .u-picker__title {
595
- color: $u-content-color;
596
- }
597
-
598
- .u-picker-body {
599
- width: 100%;
600
- height: 500rpx;
601
- overflow: hidden;
602
- background-color: #fff;
603
- }
604
-
605
- .u-column-item {
606
- @include vue-flex;
607
- align-items: center;
608
- justify-content: center;
609
- font-size: 32rpx;
610
- color: $u-main-color;
611
- padding: 0 8rpx;
612
- }
613
-
614
- .u-text {
615
- font-size: 24rpx;
616
- padding-left: 8rpx;
617
- }
618
-
619
- .u-btn-picker {
620
- padding: 16rpx;
621
- box-sizing: border-box;
622
- text-align: center;
623
- text-decoration: none;
624
- }
625
-
626
- .u-opacity {
627
- opacity: 0.5;
628
- }
629
-
630
- .u-btn-picker--primary {
631
- color: $u-type-primary;
632
- }
633
-
634
- .u-btn-picker--tips {
635
- color: $u-tips-color;
636
- }
637
- </style>
1
+ <template>
2
+ <u-popup
3
+ :maskCloseAble="maskCloseAble"
4
+ mode="bottom"
5
+ :popup="false"
6
+ v-model="popupValue"
7
+ length="auto"
8
+ :safeAreaInsetBottom="safeAreaInsetBottom"
9
+ @close="close"
10
+ :z-index="uZIndex"
11
+ >
12
+ <view class="u-datetime-picker">
13
+ <view class="u-picker-header" @touchmove.stop.prevent="">
14
+ <view class="u-btn-picker u-btn-picker--tips" :style="{ color: cancelColor }" hover-class="u-opacity" :hover-stay-time="150" @tap="getResult('cancel')">{{
15
+ cancelText
16
+ }}</view>
17
+ <view class="u-picker__title">{{ title }}</view>
18
+ <view
19
+ class="u-btn-picker u-btn-picker--primary"
20
+ :style="{ color: moving ? cancelColor : confirmColor }"
21
+ hover-class="u-opacity"
22
+ :hover-stay-time="150"
23
+ @touchmove.stop=""
24
+ @tap.stop="getResult('confirm')"
25
+ >
26
+ {{ confirmText }}
27
+ </view>
28
+ </view>
29
+ <view class="u-picker-body">
30
+ <picker-view v-if="mode == 'region'" :value="valueArr" @change="change" class="u-picker-view" @pickstart="pickstart" @pickend="pickend">
31
+ <picker-view-column v-if="!reset && params.province">
32
+ <view class="u-column-item" v-for="(item, index) in provinces" :key="index">
33
+ <view class="u-line-1">{{ item.label }}</view>
34
+ </view>
35
+ </picker-view-column>
36
+ <picker-view-column v-if="!reset && params.city">
37
+ <view class="u-column-item" v-for="(item, index) in citysRef" :key="index">
38
+ <view class="u-line-1">{{ item.label }}</view>
39
+ </view>
40
+ </picker-view-column>
41
+ <picker-view-column v-if="!reset && params.area">
42
+ <view class="u-column-item" v-for="(item, index) in areasRef" :key="index">
43
+ <view class="u-line-1">{{ item.label }}</view>
44
+ </view>
45
+ </picker-view-column>
46
+ </picker-view>
47
+ <picker-view v-else-if="mode == 'time'" :value="valueArr" @change="change" class="u-picker-view" @pickstart="pickstart" @pickend="pickend">
48
+ <picker-view-column v-if="!reset && params.year">
49
+ <view class="u-column-item" v-for="(item, index) in years" :key="index">
50
+ {{ item }}
51
+ <text class="u-text" v-if="showTimeTag">年</text>
52
+ </view>
53
+ </picker-view-column>
54
+ <picker-view-column v-if="!reset && params.month">
55
+ <view class="u-column-item" v-for="(item, index) in months" :key="index">
56
+ {{ formatNumber(item) }}
57
+ <text class="u-text" v-if="showTimeTag">月</text>
58
+ </view>
59
+ </picker-view-column>
60
+ <picker-view-column v-if="!reset && params.day">
61
+ <view class="u-column-item" v-for="(item, index) in days" :key="index">
62
+ {{ formatNumber(item) }}
63
+ <text class="u-text" v-if="showTimeTag">日</text>
64
+ </view>
65
+ </picker-view-column>
66
+ <picker-view-column v-if="!reset && params.hour">
67
+ <view class="u-column-item" v-for="(item, index) in hours" :key="index">
68
+ {{ formatNumber(item) }}
69
+ <text class="u-text" v-if="showTimeTag">时</text>
70
+ </view>
71
+ </picker-view-column>
72
+ <picker-view-column v-if="!reset && params.minute">
73
+ <view class="u-column-item" v-for="(item, index) in minutes" :key="index">
74
+ {{ formatNumber(item) }}
75
+ <text class="u-text" v-if="showTimeTag">分</text>
76
+ </view>
77
+ </picker-view-column>
78
+ <picker-view-column v-if="!reset && params.second">
79
+ <view class="u-column-item" v-for="(item, index) in seconds" :key="index">
80
+ {{ formatNumber(item) }}
81
+ <text class="u-text" v-if="showTimeTag">秒</text>
82
+ </view>
83
+ </picker-view-column>
84
+ </picker-view>
85
+ <picker-view v-else-if="mode == 'selector'" :value="valueArr" @change="change" class="u-picker-view" @pickstart="pickstart" @pickend="pickend">
86
+ <picker-view-column v-if="!reset">
87
+ <view class="u-column-item" v-for="(item, index) in range" :key="index">
88
+ <view class="u-line-1">{{ getItemValue(item, 'selector') }}</view>
89
+ </view>
90
+ </picker-view-column>
91
+ </picker-view>
92
+ <picker-view v-else-if="mode == 'multiSelector'" :value="valueArr" @change="change" class="u-picker-view" @pickstart="pickstart" @pickend="pickend">
93
+ <picker-view-column v-if="!reset" v-for="(item, index) in range" :key="index">
94
+ <view class="u-column-item" v-for="(item1, index1) in item" :key="index1">
95
+ <view class="u-line-1">{{ getItemValue(item1, 'multiSelector') }}</view>
96
+ </view>
97
+ </picker-view-column>
98
+ </picker-view>
99
+ </view>
100
+ </view>
101
+ </u-popup>
102
+ </template>
103
+
104
+ <script setup lang="ts">
105
+ import { ref, computed, watch, onMounted, nextTick } from 'vue';
106
+ import provinces from '../../libs/util/province';
107
+ import citys from '../../libs/util/city';
108
+ import areas from '../../libs/util/area';
109
+ import { PickerProps } from './types';
110
+
111
+ defineOptions({
112
+ name: 'u-picker'
113
+ });
114
+
115
+ /**
116
+ * picker picker弹出选择器
117
+ * @description 此选择器有两种弹出模式:一是时间模式,可以配置年,日,月,时,分,秒参数 二是地区模式,可以配置省,市,区参数
118
+ * @tutorial https://uviewpro.cn/zh/components/picker.html
119
+ * @property {Object} params 需要显示的参数,见官网说明
120
+ * @property {String} mode 模式选择,region-地区类型,time-时间类型(默认time)
121
+ * @property {String Number} start-year 可选的开始年份,mode=time时有效(默认1950)
122
+ * @property {String Number} end-year 可选的结束年份,mode=time时有效(默认2050)
123
+ * @property {Boolean} safe-area-inset-bottom 是否开启底部安全区适配(默认false)
124
+ * @property {Boolean} show-time-tag 时间模式时,是否显示后面的年月日中文提示
125
+ * @property {String} cancel-color 取消按钮的颜色(默认#606266)
126
+ * @property {String} confirm-color 确认按钮的颜色(默认#2979ff)
127
+ * @property {String} default-time 默认选中的时间,mode=time时有效
128
+ * @property {String} confirm-text 确认按钮的文字
129
+ * @property {String} cancel-text 取消按钮的文字
130
+ * @property {String} default-region 默认选中的地区,中文形式,mode=region时有效
131
+ * @property {String} default-code 默认选中的地区,编号形式,mode=region时有效
132
+ * @property {Boolean} mask-close-able 是否允许通过点击遮罩关闭Picker(默认true)
133
+ * @property {String Number} z-index 弹出时的z-index值(默认1075)
134
+ * @property {Array} default-selector 数组形式,其中每一项表示选择了range对应项中的第几个
135
+ * @property {Array} range 自定义选择的数据,mode=selector或mode=multiSelector时有效
136
+ * @property {String} range-key 当range参数的元素为对象时,指定Object中的哪个key的值作为选择器显示内容
137
+ * @event {Function} confirm 点击确定按钮,返回当前选择的值
138
+ * @event {Function} cancel 点击取消按钮,返回当前选择的值
139
+ * @example <u-picker v-model="show" mode="time"></u-picker>
140
+ */
141
+ const props = defineProps(PickerProps);
142
+
143
+ const popupValue = computed({
144
+ get: () => props.modelValue,
145
+ set: (val: boolean) => emit('update:modelValue', val)
146
+ });
147
+
148
+ const emit = defineEmits(['update:modelValue', 'confirm', 'cancel', 'columnchange']);
149
+
150
+ // 主要数据
151
+ const years = ref<number[]>([]);
152
+ const months = ref<number[]>([]);
153
+ const days = ref<number[]>([]);
154
+ const hours = ref<number[]>([]);
155
+ const minutes = ref<number[]>([]);
156
+ const seconds = ref<number[]>([]);
157
+ const year = ref<number>(0);
158
+ const month = ref<number>(0);
159
+ const day = ref<number>(0);
160
+ const hour = ref<number>(0);
161
+ const minute = ref<number>(0);
162
+ const second = ref<number>(0);
163
+ const reset = ref(false);
164
+ const startDate = ref('');
165
+ const endDate = ref('');
166
+ const valueArr = ref<number[]>([]);
167
+ const provincesRef = ref<any[]>(provinces);
168
+ const citysRef = ref<any[]>(citys[0]);
169
+ const areasRef = ref<any[]>(areas[0][0]);
170
+ const province = ref<number>(0);
171
+ const city = ref<number>(0);
172
+ const area = ref<number>(0);
173
+ // 列是否还在滑动中,微信小程序如果在滑动中就点确定,结果可能不准确
174
+ const moving = ref(false);
175
+ const multiSelectorValue = ref<number[]>([]);
176
+
177
+ // 计算属性
178
+ // 引用这几个变量,是为了监听其变化
179
+ const propsChange = computed(() => `${props.mode}-${props.defaultTime}-${props.startYear}-${props.endYear}-${props.defaultRegion}-${props.areaCode}`);
180
+ // 引用这几个变量,是为了监听其变化
181
+ const regionChange = computed(() => `${province.value}-${city.value}`);
182
+
183
+ const yearAndMonth = computed(() => `${year.value}-${month.value}`);
184
+ // 如果用户有传递z-index值,优先使用
185
+ const uZIndex = computed(() => (props.zIndex ? props.zIndex : 1075));
186
+
187
+ // 监听
188
+ watch(propsChange, () => {
189
+ reset.value = true;
190
+ setTimeout(() => init(), 10);
191
+ });
192
+
193
+ // 如果地区发生变化,为了让picker联动起来,必须重置this.citys和this.areas
194
+ watch(regionChange, () => {
195
+ citysRef.value = citys[province.value];
196
+ areasRef.value = areas[province.value][city.value];
197
+ });
198
+
199
+ // watch监听月份的变化,实时变更日的天数,因为不同月份,天数不一样
200
+ // 一个月可能有30,31天,甚至闰年2月的29天,平年2月28天
201
+ watch(yearAndMonth, () => {
202
+ if (props.params.year) setDays();
203
+ });
204
+ watch(
205
+ () => props.modelValue,
206
+ n => {
207
+ if (n) {
208
+ reset.value = true;
209
+ setTimeout(() => init(), 10);
210
+ }
211
+ }
212
+ );
213
+
214
+ // 方法区
215
+ /**
216
+ * 标识滑动开始,只有微信小程序才有这样的事件
217
+ */
218
+ function pickstart() {
219
+ // #ifdef MP-WEIXIN
220
+ moving.value = true;
221
+ // #endif
222
+ }
223
+ /**
224
+ * 标识滑动结束
225
+ */
226
+ function pickend() {
227
+ // #ifdef MP-WEIXIN
228
+ moving.value = false;
229
+ // #endif
230
+ }
231
+ /**
232
+ * 对单列和多列形式的判断是否有传入变量的情况
233
+ * @param item 当前项
234
+ * @param mode 模式
235
+ */
236
+ function getItemValue(item: any, mode: string) {
237
+ // 目前(2020-05-25)uni-app对微信小程序编译有错误,导致v-if为false中的内容也执行,错误导致
238
+ // 单列模式或者多列模式中的getItemValue同时被执行,故在这里再加一层判断
239
+ if (props.mode == mode) {
240
+ return typeof item == 'object' ? item[props.rangeKey] : item;
241
+ }
242
+ }
243
+ /**
244
+ * 小于10前面补0,用于月份,日期,时分秒等
245
+ */
246
+ function formatNumber(num: number | string) {
247
+ return +num < 10 ? '0' + num : String(num);
248
+ }
249
+ /**
250
+ * 生成递进的数组
251
+ */
252
+ function generateArray(start: number, end: number) {
253
+ // 转为数值格式,否则用户给end-year等传递字符串值时,下面的end+1会导致字符串拼接,而不是相加
254
+ start = Number(start);
255
+ end = Number(end);
256
+ end = end > start ? end : start;
257
+ // 生成数组,获取其中的索引,并剪出来
258
+ return [...Array(end + 1).keys()].slice(start);
259
+ }
260
+ /**
261
+ * 获取数组中指定值的索引
262
+ */
263
+ function getIndex(arr: any[], val: any) {
264
+ let index = arr.indexOf(val);
265
+ // 如果index为-1(即找不到index值),~(-1)=-(-1)-1=0,导致条件不成立
266
+ return ~index ? index : 0;
267
+ }
268
+ /**
269
+ * 日期时间处理,初始化各时间字段
270
+ */
271
+ function initTimeValue() {
272
+ // 格式化时间,在IE浏览器(uni不存在此情况),无法识别日期间的"-"间隔符号
273
+ let fdate = props.defaultTime.replace(/\-/g, '/');
274
+ fdate = fdate && fdate.indexOf('/') == -1 ? `2020/01/01 ${fdate}` : fdate;
275
+ let time: Date;
276
+ if (fdate) time = new Date(fdate);
277
+ else time = new Date();
278
+ // 获取年日月时分秒
279
+ year.value = time.getFullYear();
280
+ month.value = Number(time.getMonth()) + 1;
281
+ day.value = time.getDate();
282
+ hour.value = time.getHours();
283
+ minute.value = time.getMinutes();
284
+ second.value = time.getSeconds();
285
+ }
286
+ /**
287
+ * 初始化picker各列数据
288
+ */
289
+ function init() {
290
+ valueArr.value = [];
291
+ reset.value = false;
292
+ if (props.mode == 'time') {
293
+ initTimeValue();
294
+ if (props.params.year) {
295
+ valueArr.value.push(0);
296
+ setYears();
297
+ }
298
+ if (props.params.month) {
299
+ valueArr.value.push(0);
300
+ setMonths();
301
+ }
302
+ if (props.params.day) {
303
+ valueArr.value.push(0);
304
+ setDays();
305
+ }
306
+ if (props.params.hour) {
307
+ valueArr.value.push(0);
308
+ setHours();
309
+ }
310
+ if (props.params.minute) {
311
+ valueArr.value.push(0);
312
+ setMinutes();
313
+ }
314
+ if (props.params.second) {
315
+ valueArr.value.push(0);
316
+ setSeconds();
317
+ }
318
+ } else if (props.mode == 'region') {
319
+ if (props.params.province) {
320
+ valueArr.value.push(0);
321
+ setProvinces();
322
+ }
323
+ if (props.params.city) {
324
+ valueArr.value.push(0);
325
+ setCitys();
326
+ }
327
+ if (props.params.area) {
328
+ valueArr.value.push(0);
329
+ setAreas();
330
+ }
331
+ } else if (props.mode == 'selector') {
332
+ valueArr.value = props.defaultSelector as number[];
333
+ } else if (props.mode == 'multiSelector') {
334
+ valueArr.value = props.defaultSelector as number[];
335
+ multiSelectorValue.value = props.defaultSelector as number[];
336
+ }
337
+ nextTick(() => {});
338
+ }
339
+ /**
340
+ * 设置年份列
341
+ */
342
+ function setYears() {
343
+ // 获取年份集合
344
+ years.value = generateArray(props.startYear as number, props.endYear as number);
345
+ // 设置this.valueArr某一项的值,是为了让picker预选中某一个值
346
+ valueArr.value.splice(valueArr.value.length - 1, 1, getIndex(years.value, year.value));
347
+ }
348
+ /**
349
+ * 设置月份列
350
+ */
351
+ function setMonths() {
352
+ months.value = generateArray(1, 12);
353
+ valueArr.value.splice(valueArr.value.length - 1, 1, getIndex(months.value, month.value));
354
+ }
355
+ /**
356
+ * 设置天数列
357
+ */
358
+ function setDays() {
359
+ let totalDays = new Date(year.value, month.value, 0).getDate();
360
+ days.value = generateArray(1, totalDays);
361
+ let index = 0;
362
+ // 这里不能使用类似setMonths()中的this.valueArr.splice(this.valueArr.length - 1, 1, xxx)做法
363
+ // 因为this.month和this.year变化时,会触发watch中的this.setDays(),导致this.valueArr.length计算有误
364
+ if (props.params.year && props.params.month) index = 2;
365
+ else if (props.params.month) index = 1;
366
+ else if (props.params.year) index = 1;
367
+ else index = 0;
368
+ // 当月份变化时,会导致日期的天数也会变化,如果原来选的天数大于变化后的天数,则重置为变化后的最大值
369
+ // 比如原来选中3月31日,调整为2月后,日期变为最大29,这时如果day值继续为31显然不合理,于是将其置为29(picker-column从1开始)
370
+ if (day.value > days.value.length) day.value = days.value.length;
371
+ valueArr.value.splice(index, 1, getIndex(days.value, day.value));
372
+ }
373
+ /**
374
+ * 设置小时列
375
+ */
376
+ function setHours() {
377
+ hours.value = generateArray(0, 23);
378
+ valueArr.value.splice(valueArr.value.length - 1, 1, getIndex(hours.value, hour.value));
379
+ }
380
+ /**
381
+ * 设置分钟列
382
+ */
383
+ function setMinutes() {
384
+ minutes.value = generateArray(0, 59);
385
+ valueArr.value.splice(valueArr.value.length - 1, 1, getIndex(minutes.value, minute.value));
386
+ }
387
+ /**
388
+ * 设置秒数列
389
+ */
390
+ function setSeconds() {
391
+ seconds.value = generateArray(0, 59);
392
+ valueArr.value.splice(valueArr.value.length - 1, 1, getIndex(seconds.value, second.value));
393
+ }
394
+ /**
395
+ * 设置省份列
396
+ */
397
+ function setProvinces() {
398
+ // 判断是否需要province参数
399
+ if (!props.params.province) return;
400
+ let tmp: any = '';
401
+ let useCode = false;
402
+ // 如果同时配置了defaultRegion和areaCode,优先使用areaCode参数
403
+ if (props.areaCode.length) {
404
+ tmp = props.areaCode[0];
405
+ useCode = true;
406
+ } else if (props.defaultRegion.length) tmp = props.defaultRegion[0];
407
+ else tmp = 0;
408
+ // 历遍省份数组匹配
409
+ provinces.map((v: any, k: number) => {
410
+ if (useCode ? v.value == tmp : v.label == tmp) {
411
+ tmp = k;
412
+ }
413
+ });
414
+ province.value = tmp;
415
+ provincesRef.value = provinces;
416
+ // 设置默认省份的值
417
+ valueArr.value.splice(0, 1, province.value);
418
+ }
419
+ /**
420
+ * 设置城市列
421
+ */
422
+ function setCitys() {
423
+ if (!props.params.city) return;
424
+ let tmp: any = '';
425
+ let useCode = false;
426
+ if (props.areaCode.length) {
427
+ tmp = props.areaCode[1];
428
+ useCode = true;
429
+ } else if (props.defaultRegion.length) tmp = props.defaultRegion[1];
430
+ else tmp = 0;
431
+ // 历遍城市数组匹配
432
+ citys[province.value].map((v: any, k: number) => {
433
+ if (useCode ? v.value == tmp : v.label == tmp) {
434
+ tmp = k;
435
+ }
436
+ });
437
+ city.value = tmp;
438
+ citysRef.value = citys[province.value];
439
+ // 设置默认城市的值
440
+ valueArr.value.splice(1, 1, city.value);
441
+ }
442
+ /**
443
+ * 设置区县列
444
+ */
445
+ function setAreas() {
446
+ if (!props.params.area) return;
447
+ let tmp: any = '';
448
+ let useCode = false;
449
+ if (props.areaCode.length) {
450
+ tmp = props.areaCode[2];
451
+ useCode = true;
452
+ } else if (props.defaultRegion.length) tmp = props.defaultRegion[2];
453
+ else tmp = 0;
454
+ // 历遍区县数组匹配
455
+ areas[province.value][city.value].map((v: any, k: number) => {
456
+ if (useCode ? v.value == tmp : v.label == tmp) {
457
+ tmp = k;
458
+ }
459
+ });
460
+ area.value = tmp;
461
+ areasRef.value = areas[province.value][city.value];
462
+ // 设置默认区县的值
463
+ valueArr.value.splice(2, 1, area.value);
464
+ }
465
+ /**
466
+ * 关闭picker弹窗
467
+ */
468
+ function close() {
469
+ emit('update:modelValue', false);
470
+ }
471
+ /**
472
+ * 用户更改picker的列选项
473
+ * @param e 事件对象
474
+ */
475
+ function change(e: any) {
476
+ valueArr.value = e.detail.value;
477
+ let i = 0;
478
+ if (props.mode == 'time') {
479
+ // 这里使用i++,是因为valueArr数组的长度是不确定长度的,它根据params的值来配置长度
480
+ // 进入if规则,i会加1,保证了能获取准确的值
481
+ if (props.params.year) year.value = years.value[valueArr.value[i++]];
482
+ if (props.params.month) month.value = months.value[valueArr.value[i++]];
483
+ if (props.params.day) day.value = days.value[valueArr.value[i++]];
484
+ if (props.params.hour) hour.value = hours.value[valueArr.value[i++]];
485
+ if (props.params.minute) minute.value = minutes.value[valueArr.value[i++]];
486
+ if (props.params.second) second.value = seconds.value[valueArr.value[i++]];
487
+ } else if (props.mode == 'region') {
488
+ if (props.params.province) province.value = valueArr.value[i++];
489
+ if (props.params.city) city.value = valueArr.value[i++];
490
+ if (props.params.area) area.value = valueArr.value[i++];
491
+ } else if (props.mode == 'multiSelector') {
492
+ let index: number | null = null;
493
+ // 对比前后两个数组,寻找变更的是哪一列,如果某一个元素不同,即可判定该列发生了变化
494
+ (props.defaultSelector as number[]).map((val: number, idx: number) => {
495
+ if (val != e.detail.value[idx]) index = idx;
496
+ });
497
+ // 为了让用户对多列变化时,对动态设置其他列的变更
498
+ if (index != null) {
499
+ emit('columnchange', {
500
+ column: index,
501
+ index: e.detail.value[index]
502
+ });
503
+ }
504
+ }
505
+ }
506
+ /**
507
+ * 用户点击确定/取消按钮,获取结果
508
+ * @param event 事件名
509
+ */
510
+ function getResult(event: string | null = null) {
511
+ // #ifdef MP-WEIXIN
512
+ // 微信小程序滑动中不允许点击确定,防止取值不准确
513
+ if (moving.value) return;
514
+ // #endif
515
+ let result: any = {};
516
+ // 只返回用户在params中配置了为true的字段
517
+ if (props.mode == 'time') {
518
+ if (props.params.year) result.year = formatNumber(year.value || 0); // 年
519
+ if (props.params.month) result.month = formatNumber(month.value || 0); // 月
520
+ if (props.params.day) result.day = formatNumber(day.value || 0); // 日
521
+ if (props.params.hour) result.hour = formatNumber(hour.value || 0); // 时
522
+ if (props.params.minute) result.minute = formatNumber(minute.value || 0); // 分
523
+ if (props.params.second) result.second = formatNumber(second.value || 0); // 秒
524
+ if (props.params.timestamp) result.timestamp = getTimestamp(); // 时间戳
525
+ } else if (props.mode == 'region') {
526
+ // 地区模式,返回省市区对象
527
+ if (props.params.province) result.province = provinces[province.value];
528
+ if (props.params.city) result.city = citys[province.value][city.value];
529
+ if (props.params.area) result.area = areas[province.value][city.value][area.value];
530
+ } else if (props.mode == 'selector') {
531
+ // 单列选择器,直接返回选中下标数组
532
+ result = valueArr.value;
533
+ } else if (props.mode == 'multiSelector') {
534
+ // 多列选择器,直接返回选中下标数组
535
+ result = valueArr.value;
536
+ }
537
+ // 只允许 emit 已声明的事件类型
538
+ if (event && ['update:modelValue', 'confirm', 'cancel', 'columnchange'].includes(event)) emit(event as 'update:modelValue' | 'confirm' | 'cancel' | 'columnchange', result);
539
+ close();
540
+ }
541
+ /**
542
+ * 获取时间戳(秒)
543
+ * @returns {number} 时间戳
544
+ */
545
+ function getTimestamp() {
546
+ // yyyy-mm-dd为安卓写法,不支持iOS,需要使用"/"分隔,才能二者兼容
547
+ let time = year.value + '/' + month.value + '/' + day.value + ' ' + hour.value + ':' + minute.value + ':' + second.value;
548
+ return new Date(time).getTime() / 1000;
549
+ }
550
+
551
+ // 组件挂载时初始化
552
+ onMounted(() => {
553
+ init();
554
+ });
555
+ </script>
556
+
557
+ <style lang="scss" scoped>
558
+ @import '../../libs/css/style.components.scss';
559
+
560
+ .u-datetime-picker {
561
+ position: relative;
562
+ z-index: 999;
563
+ }
564
+
565
+ .u-picker-view {
566
+ height: 100%;
567
+ box-sizing: border-box;
568
+ }
569
+
570
+ .u-picker-header {
571
+ width: 100%;
572
+ height: 90rpx;
573
+ padding: 0 40rpx;
574
+ @include vue-flex;
575
+ justify-content: space-between;
576
+ align-items: center;
577
+ box-sizing: border-box;
578
+ font-size: 30rpx;
579
+ background: #fff;
580
+ position: relative;
581
+ }
582
+
583
+ .u-picker-header::after {
584
+ content: '';
585
+ position: absolute;
586
+ border-bottom: 1rpx solid #eaeef1;
587
+ -webkit-transform: scaleY(0.5);
588
+ transform: scaleY(0.5);
589
+ bottom: 0;
590
+ right: 0;
591
+ left: 0;
592
+ }
593
+
594
+ .u-picker__title {
595
+ color: $u-content-color;
596
+ }
597
+
598
+ .u-picker-body {
599
+ width: 100%;
600
+ height: 500rpx;
601
+ overflow: hidden;
602
+ background-color: #fff;
603
+ }
604
+
605
+ .u-column-item {
606
+ @include vue-flex;
607
+ align-items: center;
608
+ justify-content: center;
609
+ font-size: 32rpx;
610
+ color: $u-main-color;
611
+ padding: 0 8rpx;
612
+ }
613
+
614
+ .u-text {
615
+ font-size: 24rpx;
616
+ padding-left: 8rpx;
617
+ }
618
+
619
+ .u-btn-picker {
620
+ padding: 16rpx;
621
+ box-sizing: border-box;
622
+ text-align: center;
623
+ text-decoration: none;
624
+ }
625
+
626
+ .u-opacity {
627
+ opacity: 0.5;
628
+ }
629
+
630
+ .u-btn-picker--primary {
631
+ color: $u-type-primary;
632
+ }
633
+
634
+ .u-btn-picker--tips {
635
+ color: $u-tips-color;
636
+ }
637
+ </style>