uview-pro 0.3.0 → 0.3.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 (249) hide show
  1. package/changelog.md +603 -574
  2. package/components/common/props.ts +22 -22
  3. package/components/u-action-sheet/types.ts +37 -37
  4. package/components/u-action-sheet/u-action-sheet.vue +178 -178
  5. package/components/u-alert-tips/types.ts +41 -41
  6. package/components/u-alert-tips/u-alert-tips.vue +238 -238
  7. package/components/u-avatar/types.ts +36 -36
  8. package/components/u-avatar/u-avatar.vue +217 -217
  9. package/components/u-avatar-cropper/types.ts +23 -23
  10. package/components/u-avatar-cropper/u-avatar-cropper.vue +297 -297
  11. package/components/u-avatar-cropper/weCropper.d.ts +62 -62
  12. package/components/u-avatar-cropper/weCropper.js +1281 -1281
  13. package/components/u-avatar-cropper/weCropper.ts +1276 -1276
  14. package/components/u-back-top/types.ts +36 -36
  15. package/components/u-back-top/u-back-top.vue +140 -140
  16. package/components/u-badge/types.ts +38 -38
  17. package/components/u-badge/u-badge.vue +183 -183
  18. package/components/u-button/types.ts +66 -66
  19. package/components/u-button/u-button.vue +579 -579
  20. package/components/u-calendar/types.ts +75 -75
  21. package/components/u-calendar/u-calendar.vue +793 -793
  22. package/components/u-car-keyboard/types.ts +14 -14
  23. package/components/u-car-keyboard/u-car-keyboard.vue +262 -262
  24. package/components/u-card/types.ts +61 -61
  25. package/components/u-card/u-card.vue +209 -209
  26. package/components/u-cell-group/types.ts +19 -19
  27. package/components/u-cell-group/u-cell-group.vue +60 -60
  28. package/components/u-cell-item/types.ts +56 -56
  29. package/components/u-cell-item/u-cell-item.vue +226 -226
  30. package/components/u-checkbox/types.ts +33 -33
  31. package/components/u-checkbox/u-checkbox.vue +282 -282
  32. package/components/u-checkbox-group/types.ts +34 -34
  33. package/components/u-checkbox-group/u-checkbox-group.vue +130 -130
  34. package/components/u-circle-progress/types.ts +54 -54
  35. package/components/u-circle-progress/u-circle-progress.vue +206 -206
  36. package/components/u-city-select/types.ts +22 -22
  37. package/components/u-city-select/u-city-select.vue +276 -276
  38. package/components/u-col/types.ts +32 -32
  39. package/components/u-col/u-col.vue +142 -142
  40. package/components/u-collapse/types.ts +33 -33
  41. package/components/u-collapse/u-collapse.vue +190 -190
  42. package/components/u-collapse-item/types.ts +27 -27
  43. package/components/u-collapse-item/u-collapse-item.vue +290 -290
  44. package/components/u-column-notice/types.ts +50 -50
  45. package/components/u-column-notice/u-column-notice.vue +222 -222
  46. package/components/u-count-down/types.ts +44 -44
  47. package/components/u-count-down/u-count-down.vue +286 -286
  48. package/components/u-count-to/types.ts +34 -34
  49. package/components/u-count-to/u-count-to.vue +266 -266
  50. package/components/u-divider/types.ts +33 -33
  51. package/components/u-divider/u-divider.vue +145 -145
  52. package/components/u-dropdown/types.ts +34 -34
  53. package/components/u-dropdown/u-dropdown.vue +330 -330
  54. package/components/u-dropdown-item/types.ts +29 -29
  55. package/components/u-dropdown-item/u-dropdown-item.vue +120 -120
  56. package/components/u-empty/types.ts +38 -38
  57. package/components/u-empty/u-empty.vue +103 -103
  58. package/components/u-field/types.ts +71 -71
  59. package/components/u-field/u-field.vue +388 -388
  60. package/components/u-form/types.ts +29 -29
  61. package/components/u-form/u-form.vue +130 -130
  62. package/components/u-form-item/types.ts +72 -72
  63. package/components/u-form-item/u-form-item.vue +447 -447
  64. package/components/u-full-screen/types.ts +16 -16
  65. package/components/u-full-screen/u-full-screen.vue +103 -103
  66. package/components/u-gap/types.ts +20 -20
  67. package/components/u-gap/u-gap.vue +50 -50
  68. package/components/u-grid/types.ts +21 -21
  69. package/components/u-grid/u-grid.vue +91 -91
  70. package/components/u-grid-item/types.ts +16 -16
  71. package/components/u-grid-item/u-grid-item.vue +130 -130
  72. package/components/u-icon/types.ts +61 -61
  73. package/components/u-icon/u-icon.vue +296 -296
  74. package/components/u-image/types.ts +51 -51
  75. package/components/u-image/u-image.vue +239 -239
  76. package/components/u-index-anchor/types.ts +16 -16
  77. package/components/u-index-anchor/u-index-anchor.vue +120 -94
  78. package/components/u-index-list/types.ts +43 -43
  79. package/components/u-index-list/u-index-list.vue +346 -352
  80. package/components/u-input/types.ts +137 -137
  81. package/components/u-input/u-input.vue +288 -288
  82. package/components/u-keyboard/types.ts +40 -40
  83. package/components/u-keyboard/u-keyboard.vue +178 -178
  84. package/components/u-lazy-load/types.ts +37 -37
  85. package/components/u-lazy-load/u-lazy-load.vue +246 -246
  86. package/components/u-line/types.ts +44 -44
  87. package/components/u-line/u-line.vue +68 -68
  88. package/components/u-line-progress/types.ts +58 -58
  89. package/components/u-line-progress/u-line-progress.vue +126 -126
  90. package/components/u-link/types.ts +43 -43
  91. package/components/u-link/u-link.vue +84 -84
  92. package/components/u-loading/types.ts +35 -35
  93. package/components/u-loading/u-loading.vue +105 -105
  94. package/components/u-loading-popup/types.ts +26 -26
  95. package/components/u-loading-popup/u-loading-popup.vue +253 -253
  96. package/components/u-loadmore/types.ts +79 -79
  97. package/components/u-loadmore/u-loadmore.vue +156 -156
  98. package/components/u-mask/types.ts +40 -40
  99. package/components/u-mask/u-mask.vue +113 -113
  100. package/components/u-message-input/types.ts +74 -74
  101. package/components/u-message-input/u-message-input.vue +281 -281
  102. package/components/u-modal/types.ts +118 -118
  103. package/components/u-modal/u-modal.vue +220 -220
  104. package/components/u-navbar/types.ts +103 -103
  105. package/components/u-navbar/u-navbar.vue +251 -251
  106. package/components/u-no-network/image.ts +2 -2
  107. package/components/u-no-network/types.ts +28 -28
  108. package/components/u-no-network/u-no-network.vue +303 -303
  109. package/components/u-notice-bar/types.ts +111 -111
  110. package/components/u-notice-bar/u-notice-bar.vue +189 -189
  111. package/components/u-number-box/types.ts +42 -42
  112. package/components/u-number-box/u-number-box.vue +321 -321
  113. package/components/u-number-keyboard/types.ts +26 -26
  114. package/components/u-number-keyboard/u-number-keyboard.vue +188 -188
  115. package/components/u-picker/types.ts +123 -123
  116. package/components/u-picker/u-picker.vue +685 -685
  117. package/components/u-popup/types.ts +59 -59
  118. package/components/u-popup/u-popup.vue +385 -385
  119. package/components/u-radio/types.ts +27 -27
  120. package/components/u-radio/u-radio.vue +279 -279
  121. package/components/u-radio-group/types.ts +31 -31
  122. package/components/u-radio-group/u-radio-group.vue +96 -96
  123. package/components/u-rate/types.ts +42 -42
  124. package/components/u-rate/u-rate.vue +249 -249
  125. package/components/u-read-more/types.ts +37 -37
  126. package/components/u-read-more/u-read-more.vue +172 -172
  127. package/components/u-root-portal/u-root-portal.vue +56 -56
  128. package/components/u-row/types.ts +22 -22
  129. package/components/u-row/u-row.vue +105 -105
  130. package/components/u-row-notice/types.ts +41 -41
  131. package/components/u-row-notice/u-row-notice.vue +256 -256
  132. package/components/u-safe-bottom/u-safe-bottom.vue +57 -57
  133. package/components/u-search/types.ts +55 -55
  134. package/components/u-search/u-search.vue +279 -279
  135. package/components/u-section/types.ts +34 -34
  136. package/components/u-section/u-section.vue +150 -150
  137. package/components/u-select/types.ts +45 -45
  138. package/components/u-select/u-select.vue +388 -388
  139. package/components/u-skeleton/types.ts +22 -22
  140. package/components/u-skeleton/u-skeleton.vue +231 -231
  141. package/components/u-slider/types.ts +34 -34
  142. package/components/u-slider/u-slider.vue +255 -255
  143. package/components/u-status-bar/u-status-bar.vue +74 -74
  144. package/components/u-steps/types.ts +30 -30
  145. package/components/u-steps/u-steps.vue +181 -181
  146. package/components/u-sticky/types.ts +24 -24
  147. package/components/u-sticky/u-sticky.vue +178 -178
  148. package/components/u-subsection/types.ts +38 -38
  149. package/components/u-subsection/u-subsection.vue +339 -339
  150. package/components/u-swipe-action/types.ts +52 -52
  151. package/components/u-swipe-action/u-swipe-action.vue +276 -276
  152. package/components/u-swiper/types.ts +49 -49
  153. package/components/u-swiper/u-swiper.vue +308 -308
  154. package/components/u-switch/types.ts +30 -30
  155. package/components/u-switch/u-switch.vue +150 -150
  156. package/components/u-tabbar/types.ts +38 -38
  157. package/components/u-tabbar/u-tabbar.vue +315 -315
  158. package/components/u-table/types.ts +27 -27
  159. package/components/u-table/u-table.vue +67 -67
  160. package/components/u-tabs/types.ts +53 -53
  161. package/components/u-tabs/u-tabs.vue +302 -302
  162. package/components/u-tabs-swiper/types.ts +55 -55
  163. package/components/u-tabs-swiper/u-tabs-swiper.vue +409 -409
  164. package/components/u-tag/types.ts +39 -39
  165. package/components/u-tag/u-tag.vue +268 -268
  166. package/components/u-td/types.ts +14 -14
  167. package/components/u-td/u-td.vue +98 -98
  168. package/components/u-text/types.ts +72 -72
  169. package/components/u-text/u-text.vue +343 -343
  170. package/components/u-th/types.ts +14 -14
  171. package/components/u-th/u-th.vue +92 -92
  172. package/components/u-time-line/u-time-line.vue +53 -53
  173. package/components/u-time-line-item/types.ts +16 -16
  174. package/components/u-time-line-item/u-time-line-item.vue +90 -90
  175. package/components/u-toast/types.ts +38 -38
  176. package/components/u-toast/u-toast.vue +240 -240
  177. package/components/u-top-tips/types.ts +16 -16
  178. package/components/u-top-tips/u-top-tips.vue +130 -130
  179. package/components/u-tr/types.ts +11 -11
  180. package/components/u-tr/u-tr.vue +39 -39
  181. package/components/u-upload/types.ts +82 -82
  182. package/components/u-upload/u-upload.vue +568 -568
  183. package/components/u-verification-code/types.ts +24 -24
  184. package/components/u-verification-code/u-verification-code.vue +176 -176
  185. package/components/u-waterfall/types.ts +18 -18
  186. package/components/u-waterfall/u-waterfall.vue +187 -187
  187. package/iconfont.css +913 -913
  188. package/index.scss +25 -25
  189. package/index.ts +38 -38
  190. package/libs/config/config.ts +26 -26
  191. package/libs/config/zIndex.ts +37 -37
  192. package/libs/css/color.scss +155 -155
  193. package/libs/css/common.scss +178 -178
  194. package/libs/css/style.components.scss +16 -16
  195. package/libs/css/style.h5.scss +8 -8
  196. package/libs/css/style.mp.scss +72 -72
  197. package/libs/css/style.nvue.scss +15 -15
  198. package/libs/css/style.vue.scss +188 -188
  199. package/libs/function/$parent.ts +24 -24
  200. package/libs/function/addUnit.ts +13 -13
  201. package/libs/function/color.ts +37 -37
  202. package/libs/function/colorGradient.ts +139 -139
  203. package/libs/function/debounce.ts +28 -28
  204. package/libs/function/deepClone.ts +39 -39
  205. package/libs/function/deepMerge.ts +35 -35
  206. package/libs/function/getParent.ts +63 -63
  207. package/libs/function/getRect.ts +26 -26
  208. package/libs/function/guid.ts +42 -42
  209. package/libs/function/md5.ts +403 -403
  210. package/libs/function/parent.ts +21 -21
  211. package/libs/function/queryParams.ts +64 -64
  212. package/libs/function/random.ts +16 -16
  213. package/libs/function/randomArray.ts +11 -11
  214. package/libs/function/route.ts +118 -118
  215. package/libs/function/styleUtils.ts +83 -83
  216. package/libs/function/sys.ts +15 -15
  217. package/libs/function/test.ts +289 -289
  218. package/libs/function/throttle.ts +31 -31
  219. package/libs/function/timeFormat.ts +55 -55
  220. package/libs/function/timeFrom.ts +48 -48
  221. package/libs/function/toast.ts +14 -14
  222. package/libs/function/trim.ts +21 -21
  223. package/libs/function/type2icon.ts +39 -39
  224. package/libs/hooks/index.ts +4 -4
  225. package/libs/hooks/useCompRelation.ts +383 -364
  226. package/libs/hooks/useComponent.ts +759 -759
  227. package/libs/hooks/useEmitter.ts +79 -79
  228. package/libs/hooks/useParent.ts +33 -33
  229. package/libs/hooks/useRect.ts +33 -33
  230. package/libs/index.ts +337 -337
  231. package/libs/request/auto-http.ts +76 -76
  232. package/libs/request/index.ts +242 -242
  233. package/libs/store/index.ts +88 -88
  234. package/libs/util/async-validator.d.ts +75 -75
  235. package/libs/util/async-validator.js +1 -1
  236. package/libs/util/calendar.d.ts +57 -57
  237. package/libs/util/emitter.ts +112 -112
  238. package/libs/util/eventBus.ts +86 -86
  239. package/libs/util/logger.ts +364 -364
  240. package/libs/util/mitt.ts +118 -118
  241. package/libs/util/parent.ts +20 -20
  242. package/package.json +4 -4
  243. package/readme.md +241 -241
  244. package/theme.scss +38 -38
  245. package/types/components.d.ts +97 -97
  246. package/types/global.d.ts +331 -331
  247. package/types/ignore-errors.d.ts +30 -30
  248. package/types/index.d.ts +19 -19
  249. package/types/uni-app.d.ts +63 -63
@@ -1,302 +1,302 @@
1
- <template>
2
- <view class="u-tabs" :style="$u.toStyle({ background: bgColor }, customStyle)" :class="customClass">
3
- <!-- $u.getRect()对组件根节点无效,因为写了.in(this),故这里获取内层接点尺寸 -->
4
- <view>
5
- <scroll-view scroll-x class="u-scroll-view" :scroll-left="scrollLeft" scroll-with-animation>
6
- <view class="u-scroll-box" :id="id" :class="{ 'u-tabs-scroll-flex': !isScroll }">
7
- <view
8
- class="u-tab-item u-line-1"
9
- :id="'u-tab-item-' + index"
10
- v-for="(item, index) in list"
11
- :key="index"
12
- @tap="clickTab(index)"
13
- :style="tabItemStyle(index)"
14
- >
15
- <u-badge :count="item[count] || item['count'] || 0" :offset="offset" size="mini"></u-badge>
16
- {{ item[name] || item['name'] }}
17
- </view>
18
- <view v-if="showBar" class="u-tab-bar" :style="tabBarStyle"></view>
19
- </view>
20
- </scroll-view>
21
- </view>
22
- </view>
23
- </template>
24
-
25
- <script lang="ts">
26
- export default {
27
- name: 'u-tabs',
28
- options: {
29
- addGlobalClass: true,
30
- // #ifndef MP-TOUTIAO
31
- virtualHost: true,
32
- // #endif
33
- styleIsolation: 'shared'
34
- }
35
- };
36
- </script>
37
-
38
- <script setup lang="ts">
39
- import { ref, computed, watch, onMounted, nextTick, getCurrentInstance } from 'vue';
40
- import { $u } from '../..';
41
- import { TabsProps } from './types';
42
-
43
- /**
44
- * tabs 标签
45
- * @description 该组件,是一个tabs标签组件,在标签多的时候,可以配置为左右滑动,标签少的时候,可以禁止滑动。 该组件的一个特点是配置为滚动模式时,激活的tab会自动移动到组件的中间位置。
46
- * @tutorial https://uviewpro.cn/zh/components/tabs.html
47
- * @property {Boolean} is-scroll tabs是否可以左右拖动(默认true)
48
- * @property {Array} list 标签数组,元素为对象,如[{name: '推荐'}]
49
- * @property {String|Number} current 指定哪个tab为激活状态(默认0)
50
- * @property {String|Number} height 导航栏的高度,单位rpx(默认80)
51
- * @property {String|Number} font-size tab文字大小,单位rpx(默认30)
52
- * @property {String|Number} duration 滑块移动一次所需的时间,单位秒(默认0.5)
53
- * @property {String} active-color 滑块和激活tab文字的颜色(默认#2979ff)
54
- * @property {String} inactive-color tabs文字颜色(默认#303133)
55
- * @property {String|Number} bar-width 滑块宽度,单位rpx(默认40)
56
- * @property {Object} active-item-style 活动tabs item的样式,对象形式
57
- * @property {Object} bar-style 底部滑块的样式,对象形式
58
- * @property {Boolean} show-bar 是否显示底部的滑块(默认true)
59
- * @property {String|Number} bar-height 滑块高度,单位rpx(默认6)
60
- * @property {String|Number} item-width 标签的宽度(默认auto)
61
- * @property {String|Number} gutter 单个tab标签的左右内边距之和,单位rpx(默认40)
62
- * @property {String} bg-color tabs导航栏的背景颜色(默认#ffffff)
63
- * @property {String} name 组件内部读取的list参数中的属性名(tab名称),见官网说明(默认name)
64
- * @property {String} count 组件内部读取的list参数中的属性名(badge徽标数),同name属性的使用,见官网说明(默认count)
65
- * @property {Array} offset 设置badge徽标数的位置偏移,格式为 [x, y],也即设置的为top和right的值,单位rpx(默认[5, 20])
66
- * @property {Boolean} bold 激活选项的字体是否加粗(默认true)
67
- * @event {Function} change 点击标签时触发
68
- * @example <u-tabs ref="tabs" :list="list" :is-scroll="false"></u-tabs>
69
- */
70
-
71
- // props 定义
72
- const props = defineProps(TabsProps);
73
-
74
- /**
75
- * emits 定义
76
- */
77
- const emit = defineEmits(['change']);
78
- const instance = getCurrentInstance();
79
- // 滚动scroll-view的左边滚动距离
80
- const scrollLeft = ref(0);
81
- // 存放对tab菜单查询后的节点信息
82
- const tabQueryInfo = ref<any[]>([]);
83
- // 屏幕宽度,单位为px
84
- const componentWidth = ref(0);
85
- // 移动bar需要通过translateX()移动的距离
86
- const scrollBarLeft = ref(0);
87
- // 父元素(tabs组件)到屏幕左边的距离
88
- const parentLeft = ref(0);
89
- // id值
90
- const id = ref($u.guid());
91
- // 当前活动tab索引
92
- const currentIndex = ref(props.current);
93
- // 滑块第一次移动时(页面刚生成时),无需动画,否则给人怪异的感觉
94
- const barFirstTimeMove = ref(true);
95
-
96
- // 监听list变化,重置索引并重新布局
97
- // 监听tab的变化,重新计算tab菜单的布局信息,因为实际使用中菜单可能是通过
98
- // 后台获取的(如新闻app顶部的菜单),获取返回需要一定时间,所以list变化时,重新获取布局信息
99
- watch(
100
- () => props.list,
101
- (n, o) => {
102
- // list变动时,重制内部索引,否则可能导致超出数组边界的情况
103
- if (n.length !== o.length) currentIndex.value = 0;
104
- // 用$nextTick等待视图更新完毕后再计算tab的局部信息,否则可能因为tab还没生成就获取,就会有问题
105
- nextTick(() => {
106
- init();
107
- });
108
- }
109
- );
110
-
111
- // 监听current变化,自动滚动
112
- watch(
113
- () => props.current,
114
- nVal => {
115
- // 视图更新后再执行移动操作
116
- nextTick(() => {
117
- currentIndex.value = nVal;
118
- scrollByIndex();
119
- });
120
- },
121
- { immediate: true }
122
- );
123
-
124
- // 移动bar的样式
125
- const tabBarStyle = computed(() => {
126
- const style: Record<string, any> = {
127
- width: props.barWidth + 'rpx',
128
- transform: `translate(${scrollBarLeft.value}px, -100%)`,
129
- // 滑块在页面渲染后第一次滑动时,无需动画效果
130
- 'transition-duration': `${barFirstTimeMove.value ? 0 : props.duration}s`,
131
- 'background-color': props.activeColor,
132
- height: props.barHeight + 'rpx',
133
- opacity: barFirstTimeMove.value ? 0 : 1,
134
- // 设置一个很大的值,它会自动取能用的最大值,不用高度的一半,是因为高度可能是单数,会有小数出现
135
- 'border-radius': `${Number(props.barHeight) / 2}px`
136
- };
137
- Object.assign(style, props.barStyle);
138
- return style;
139
- });
140
-
141
- // tab的样式
142
- function tabItemStyle(index: number) {
143
- let style: Record<string, any> = {
144
- height: props.height + 'rpx',
145
- 'line-height': props.height + 'rpx',
146
- 'font-size': props.fontSize + 'rpx',
147
- 'transition-duration': `${props.duration}s`,
148
- padding: props.isScroll ? `0 ${props.gutter}rpx` : '',
149
- flex: props.isScroll ? 'auto' : '1',
150
- width: $u.addUnit(props.itemWidth)
151
- };
152
- // 字体加粗
153
- if (index == Number(currentIndex.value) && props.bold) style.fontWeight = 'bold';
154
- if (index == Number(currentIndex.value)) {
155
- style.color = props.activeColor;
156
- // 给选中的tab item添加外部自定义的样式
157
- style = Object.assign(style, props.activeItemStyle);
158
- } else {
159
- style.color = props.inactiveColor;
160
- }
161
- return style;
162
- }
163
-
164
- /**
165
- * 初始化tab布局信息
166
- */
167
- async function init() {
168
- // 获取tabs组件的尺寸信息
169
- const tabRect = await $u.getRect('#' + id.value, instance);
170
- // tabs组件距离屏幕左边的宽度
171
- parentLeft.value = tabRect.left;
172
- // tabs组件的宽度
173
- componentWidth.value = tabRect.width;
174
- getTabRect();
175
- }
176
-
177
- /**
178
- * 点击某一个tab菜单
179
- */
180
- function clickTab(index: number) {
181
- // 点击当前活动tab,不触发事件
182
- if (index == currentIndex.value) return;
183
- // 发送事件给父组件
184
- emit('change', index);
185
- }
186
-
187
- /**
188
- * 查询tab的布局信息
189
- */
190
- function getTabRect() {
191
- // 创建节点查询
192
- const query = uni.createSelectorQuery().in(instance?.proxy);
193
- // 历遍所有tab,这里是执行了查询,最终使用exec()会一次性返回查询的数组结果
194
- for (let i = 0; i < props.list.length; i++) {
195
- // 只要size和rect两个参数
196
- query.select(`#u-tab-item-${i}`).fields({ size: true, rect: true }, () => {});
197
- }
198
- // 执行查询,一次性获取多个结果
199
- query.exec((res: any[]) => {
200
- tabQueryInfo.value = res;
201
- // 初始化滚动条和移动bar的位置
202
- scrollByIndex();
203
- });
204
- }
205
-
206
- /**
207
- * 滚动scroll-view,让活动的tab处于屏幕的中间位置
208
- */
209
- function scrollByIndex() {
210
- // 当前活动tab的布局信息,有tab菜单的width和left(为元素左边界到父元素左边界的距离)等信息
211
- const tabInfo = tabQueryInfo.value[Number(currentIndex.value)];
212
- if (!tabInfo) return;
213
- // 活动tab的宽度
214
- const tabWidth = tabInfo.width;
215
- // 活动item的左边到tabs组件左边的距离,用item的left减去tabs的left
216
- const offsetLeft = tabInfo.left - parentLeft.value;
217
- // 将活动的tabs-item移动到屏幕正中间,实际上是对scroll-view的移动
218
- const scrollL = offsetLeft - (componentWidth.value - tabWidth) / 2;
219
- scrollLeft.value = scrollL < 0 ? 0 : scrollL;
220
- // 当前活动item的中点点到左边的距离减去滑块宽度的一半,即可得到滑块所需的移动距离
221
- const left = tabInfo.left + tabInfo.width / 2 - parentLeft.value;
222
- // 计算当前活跃item到组件左边的距离
223
- scrollBarLeft.value = left - uni.upx2px(Number(props.barWidth)) / 2;
224
- // 第一次移动滑块的时候,barFirstTimeMove为true,放到延时中将其设置false
225
- // 延时是因为scrollBarLeft作用于computed计算时,需要一个过程需,否则导致出错
226
- if (barFirstTimeMove.value) {
227
- setTimeout(() => {
228
- barFirstTimeMove.value = false;
229
- }, 100);
230
- }
231
- }
232
-
233
- onMounted(() => {
234
- init();
235
- });
236
-
237
- defineExpose({ init, clickTab, scrollByIndex });
238
- </script>
239
-
240
- <style lang="scss" scoped>
241
- @import '../../libs/css/style.components.scss';
242
-
243
- view,
244
- scroll-view {
245
- box-sizing: border-box;
246
- }
247
-
248
- /* #ifndef APP-NVUE */
249
- ::-webkit-scrollbar,
250
- ::-webkit-scrollbar,
251
- ::-webkit-scrollbar {
252
- display: none;
253
- width: 0 !important;
254
- height: 0 !important;
255
- -webkit-appearance: none;
256
- background: transparent;
257
- }
258
- /* #endif */
259
-
260
- .u-scroll-box {
261
- position: relative;
262
- /* #ifdef MP-TOUTIAO */
263
- white-space: nowrap;
264
- /* #endif */
265
- }
266
-
267
- /* #ifdef H5 */
268
- // 通过样式穿透,隐藏H5下,scroll-view下的滚动条
269
- scroll-view ::v-deep ::-webkit-scrollbar {
270
- display: none;
271
- width: 0 !important;
272
- height: 0 !important;
273
- -webkit-appearance: none;
274
- background: transparent;
275
- }
276
- /* #endif */
277
-
278
- .u-scroll-view {
279
- width: 100%;
280
- white-space: nowrap;
281
- position: relative;
282
- }
283
-
284
- .u-tab-item {
285
- position: relative;
286
- /* #ifndef APP-NVUE */
287
- display: inline-block;
288
- /* #endif */
289
- text-align: center;
290
- transition-property: background-color, color;
291
- }
292
-
293
- .u-tab-bar {
294
- position: absolute;
295
- bottom: 0;
296
- }
297
-
298
- .u-tabs-scroll-flex {
299
- @include vue-flex;
300
- justify-content: space-between;
301
- }
302
- </style>
1
+ <template>
2
+ <view class="u-tabs" :style="$u.toStyle({ background: bgColor }, customStyle)" :class="customClass">
3
+ <!-- $u.getRect()对组件根节点无效,因为写了.in(this),故这里获取内层接点尺寸 -->
4
+ <view>
5
+ <scroll-view scroll-x class="u-scroll-view" :scroll-left="scrollLeft" scroll-with-animation>
6
+ <view class="u-scroll-box" :id="id" :class="{ 'u-tabs-scroll-flex': !isScroll }">
7
+ <view
8
+ class="u-tab-item u-line-1"
9
+ :id="'u-tab-item-' + index"
10
+ v-for="(item, index) in list"
11
+ :key="index"
12
+ @tap="clickTab(index)"
13
+ :style="tabItemStyle(index)"
14
+ >
15
+ <u-badge :count="item[count] || item['count'] || 0" :offset="offset" size="mini"></u-badge>
16
+ {{ item[name] || item['name'] }}
17
+ </view>
18
+ <view v-if="showBar" class="u-tab-bar" :style="tabBarStyle"></view>
19
+ </view>
20
+ </scroll-view>
21
+ </view>
22
+ </view>
23
+ </template>
24
+
25
+ <script lang="ts">
26
+ export default {
27
+ name: 'u-tabs',
28
+ options: {
29
+ addGlobalClass: true,
30
+ // #ifndef MP-TOUTIAO
31
+ virtualHost: true,
32
+ // #endif
33
+ styleIsolation: 'shared'
34
+ }
35
+ };
36
+ </script>
37
+
38
+ <script setup lang="ts">
39
+ import { ref, computed, watch, onMounted, nextTick, getCurrentInstance } from 'vue';
40
+ import { $u } from '../..';
41
+ import { TabsProps } from './types';
42
+
43
+ /**
44
+ * tabs 标签
45
+ * @description 该组件,是一个tabs标签组件,在标签多的时候,可以配置为左右滑动,标签少的时候,可以禁止滑动。 该组件的一个特点是配置为滚动模式时,激活的tab会自动移动到组件的中间位置。
46
+ * @tutorial https://uviewpro.cn/zh/components/tabs.html
47
+ * @property {Boolean} is-scroll tabs是否可以左右拖动(默认true)
48
+ * @property {Array} list 标签数组,元素为对象,如[{name: '推荐'}]
49
+ * @property {String|Number} current 指定哪个tab为激活状态(默认0)
50
+ * @property {String|Number} height 导航栏的高度,单位rpx(默认80)
51
+ * @property {String|Number} font-size tab文字大小,单位rpx(默认30)
52
+ * @property {String|Number} duration 滑块移动一次所需的时间,单位秒(默认0.5)
53
+ * @property {String} active-color 滑块和激活tab文字的颜色(默认#2979ff)
54
+ * @property {String} inactive-color tabs文字颜色(默认#303133)
55
+ * @property {String|Number} bar-width 滑块宽度,单位rpx(默认40)
56
+ * @property {Object} active-item-style 活动tabs item的样式,对象形式
57
+ * @property {Object} bar-style 底部滑块的样式,对象形式
58
+ * @property {Boolean} show-bar 是否显示底部的滑块(默认true)
59
+ * @property {String|Number} bar-height 滑块高度,单位rpx(默认6)
60
+ * @property {String|Number} item-width 标签的宽度(默认auto)
61
+ * @property {String|Number} gutter 单个tab标签的左右内边距之和,单位rpx(默认40)
62
+ * @property {String} bg-color tabs导航栏的背景颜色(默认#ffffff)
63
+ * @property {String} name 组件内部读取的list参数中的属性名(tab名称),见官网说明(默认name)
64
+ * @property {String} count 组件内部读取的list参数中的属性名(badge徽标数),同name属性的使用,见官网说明(默认count)
65
+ * @property {Array} offset 设置badge徽标数的位置偏移,格式为 [x, y],也即设置的为top和right的值,单位rpx(默认[5, 20])
66
+ * @property {Boolean} bold 激活选项的字体是否加粗(默认true)
67
+ * @event {Function} change 点击标签时触发
68
+ * @example <u-tabs ref="tabs" :list="list" :is-scroll="false"></u-tabs>
69
+ */
70
+
71
+ // props 定义
72
+ const props = defineProps(TabsProps);
73
+
74
+ /**
75
+ * emits 定义
76
+ */
77
+ const emit = defineEmits(['change']);
78
+ const instance = getCurrentInstance();
79
+ // 滚动scroll-view的左边滚动距离
80
+ const scrollLeft = ref(0);
81
+ // 存放对tab菜单查询后的节点信息
82
+ const tabQueryInfo = ref<any[]>([]);
83
+ // 屏幕宽度,单位为px
84
+ const componentWidth = ref(0);
85
+ // 移动bar需要通过translateX()移动的距离
86
+ const scrollBarLeft = ref(0);
87
+ // 父元素(tabs组件)到屏幕左边的距离
88
+ const parentLeft = ref(0);
89
+ // id值
90
+ const id = ref($u.guid());
91
+ // 当前活动tab索引
92
+ const currentIndex = ref(props.current);
93
+ // 滑块第一次移动时(页面刚生成时),无需动画,否则给人怪异的感觉
94
+ const barFirstTimeMove = ref(true);
95
+
96
+ // 监听list变化,重置索引并重新布局
97
+ // 监听tab的变化,重新计算tab菜单的布局信息,因为实际使用中菜单可能是通过
98
+ // 后台获取的(如新闻app顶部的菜单),获取返回需要一定时间,所以list变化时,重新获取布局信息
99
+ watch(
100
+ () => props.list,
101
+ (n, o) => {
102
+ // list变动时,重制内部索引,否则可能导致超出数组边界的情况
103
+ if (n.length !== o.length) currentIndex.value = 0;
104
+ // 用$nextTick等待视图更新完毕后再计算tab的局部信息,否则可能因为tab还没生成就获取,就会有问题
105
+ nextTick(() => {
106
+ init();
107
+ });
108
+ }
109
+ );
110
+
111
+ // 监听current变化,自动滚动
112
+ watch(
113
+ () => props.current,
114
+ nVal => {
115
+ // 视图更新后再执行移动操作
116
+ nextTick(() => {
117
+ currentIndex.value = nVal;
118
+ scrollByIndex();
119
+ });
120
+ },
121
+ { immediate: true }
122
+ );
123
+
124
+ // 移动bar的样式
125
+ const tabBarStyle = computed(() => {
126
+ const style: Record<string, any> = {
127
+ width: props.barWidth + 'rpx',
128
+ transform: `translate(${scrollBarLeft.value}px, -100%)`,
129
+ // 滑块在页面渲染后第一次滑动时,无需动画效果
130
+ 'transition-duration': `${barFirstTimeMove.value ? 0 : props.duration}s`,
131
+ 'background-color': props.activeColor,
132
+ height: props.barHeight + 'rpx',
133
+ opacity: barFirstTimeMove.value ? 0 : 1,
134
+ // 设置一个很大的值,它会自动取能用的最大值,不用高度的一半,是因为高度可能是单数,会有小数出现
135
+ 'border-radius': `${Number(props.barHeight) / 2}px`
136
+ };
137
+ Object.assign(style, props.barStyle);
138
+ return style;
139
+ });
140
+
141
+ // tab的样式
142
+ function tabItemStyle(index: number) {
143
+ let style: Record<string, any> = {
144
+ height: props.height + 'rpx',
145
+ 'line-height': props.height + 'rpx',
146
+ 'font-size': props.fontSize + 'rpx',
147
+ 'transition-duration': `${props.duration}s`,
148
+ padding: props.isScroll ? `0 ${props.gutter}rpx` : '',
149
+ flex: props.isScroll ? 'auto' : '1',
150
+ width: $u.addUnit(props.itemWidth)
151
+ };
152
+ // 字体加粗
153
+ if (index == Number(currentIndex.value) && props.bold) style.fontWeight = 'bold';
154
+ if (index == Number(currentIndex.value)) {
155
+ style.color = props.activeColor;
156
+ // 给选中的tab item添加外部自定义的样式
157
+ style = Object.assign(style, props.activeItemStyle);
158
+ } else {
159
+ style.color = props.inactiveColor;
160
+ }
161
+ return style;
162
+ }
163
+
164
+ /**
165
+ * 初始化tab布局信息
166
+ */
167
+ async function init() {
168
+ // 获取tabs组件的尺寸信息
169
+ const tabRect = await $u.getRect('#' + id.value, instance);
170
+ // tabs组件距离屏幕左边的宽度
171
+ parentLeft.value = tabRect.left;
172
+ // tabs组件的宽度
173
+ componentWidth.value = tabRect.width;
174
+ getTabRect();
175
+ }
176
+
177
+ /**
178
+ * 点击某一个tab菜单
179
+ */
180
+ function clickTab(index: number) {
181
+ // 点击当前活动tab,不触发事件
182
+ if (index == currentIndex.value) return;
183
+ // 发送事件给父组件
184
+ emit('change', index);
185
+ }
186
+
187
+ /**
188
+ * 查询tab的布局信息
189
+ */
190
+ function getTabRect() {
191
+ // 创建节点查询
192
+ const query = uni.createSelectorQuery().in(instance?.proxy);
193
+ // 历遍所有tab,这里是执行了查询,最终使用exec()会一次性返回查询的数组结果
194
+ for (let i = 0; i < props.list.length; i++) {
195
+ // 只要size和rect两个参数
196
+ query.select(`#u-tab-item-${i}`).fields({ size: true, rect: true }, () => {});
197
+ }
198
+ // 执行查询,一次性获取多个结果
199
+ query.exec((res: any[]) => {
200
+ tabQueryInfo.value = res;
201
+ // 初始化滚动条和移动bar的位置
202
+ scrollByIndex();
203
+ });
204
+ }
205
+
206
+ /**
207
+ * 滚动scroll-view,让活动的tab处于屏幕的中间位置
208
+ */
209
+ function scrollByIndex() {
210
+ // 当前活动tab的布局信息,有tab菜单的width和left(为元素左边界到父元素左边界的距离)等信息
211
+ const tabInfo = tabQueryInfo.value[Number(currentIndex.value)];
212
+ if (!tabInfo) return;
213
+ // 活动tab的宽度
214
+ const tabWidth = tabInfo.width;
215
+ // 活动item的左边到tabs组件左边的距离,用item的left减去tabs的left
216
+ const offsetLeft = tabInfo.left - parentLeft.value;
217
+ // 将活动的tabs-item移动到屏幕正中间,实际上是对scroll-view的移动
218
+ const scrollL = offsetLeft - (componentWidth.value - tabWidth) / 2;
219
+ scrollLeft.value = scrollL < 0 ? 0 : scrollL;
220
+ // 当前活动item的中点点到左边的距离减去滑块宽度的一半,即可得到滑块所需的移动距离
221
+ const left = tabInfo.left + tabInfo.width / 2 - parentLeft.value;
222
+ // 计算当前活跃item到组件左边的距离
223
+ scrollBarLeft.value = left - uni.upx2px(Number(props.barWidth)) / 2;
224
+ // 第一次移动滑块的时候,barFirstTimeMove为true,放到延时中将其设置false
225
+ // 延时是因为scrollBarLeft作用于computed计算时,需要一个过程需,否则导致出错
226
+ if (barFirstTimeMove.value) {
227
+ setTimeout(() => {
228
+ barFirstTimeMove.value = false;
229
+ }, 100);
230
+ }
231
+ }
232
+
233
+ onMounted(() => {
234
+ init();
235
+ });
236
+
237
+ defineExpose({ init, clickTab, scrollByIndex });
238
+ </script>
239
+
240
+ <style lang="scss" scoped>
241
+ @import '../../libs/css/style.components.scss';
242
+
243
+ view,
244
+ scroll-view {
245
+ box-sizing: border-box;
246
+ }
247
+
248
+ /* #ifndef APP-NVUE */
249
+ ::-webkit-scrollbar,
250
+ ::-webkit-scrollbar,
251
+ ::-webkit-scrollbar {
252
+ display: none;
253
+ width: 0 !important;
254
+ height: 0 !important;
255
+ -webkit-appearance: none;
256
+ background: transparent;
257
+ }
258
+ /* #endif */
259
+
260
+ .u-scroll-box {
261
+ position: relative;
262
+ /* #ifdef MP-TOUTIAO */
263
+ white-space: nowrap;
264
+ /* #endif */
265
+ }
266
+
267
+ /* #ifdef H5 */
268
+ // 通过样式穿透,隐藏H5下,scroll-view下的滚动条
269
+ scroll-view ::v-deep ::-webkit-scrollbar {
270
+ display: none;
271
+ width: 0 !important;
272
+ height: 0 !important;
273
+ -webkit-appearance: none;
274
+ background: transparent;
275
+ }
276
+ /* #endif */
277
+
278
+ .u-scroll-view {
279
+ width: 100%;
280
+ white-space: nowrap;
281
+ position: relative;
282
+ }
283
+
284
+ .u-tab-item {
285
+ position: relative;
286
+ /* #ifndef APP-NVUE */
287
+ display: inline-block;
288
+ /* #endif */
289
+ text-align: center;
290
+ transition-property: background-color, color;
291
+ }
292
+
293
+ .u-tab-bar {
294
+ position: absolute;
295
+ bottom: 0;
296
+ }
297
+
298
+ .u-tabs-scroll-flex {
299
+ @include vue-flex;
300
+ justify-content: space-between;
301
+ }
302
+ </style>