uview-pro 0.2.1 → 0.2.3

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 (246) hide show
  1. package/changelog.md +522 -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 +286 -267
  32. package/components/u-checkbox-group/types.ts +32 -32
  33. package/components/u-checkbox-group/u-checkbox-group.vue +130 -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 +195 -69
  42. package/components/u-collapse-item/types.ts +27 -27
  43. package/components/u-collapse-item/u-collapse-item.vue +311 -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 +38 -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/useComponent.ts +343 -0
  224. package/libs/hooks/useEmitter.ts +77 -77
  225. package/libs/hooks/useParent.ts +33 -31
  226. package/libs/hooks/useRect.ts +33 -33
  227. package/libs/index.ts +320 -291
  228. package/libs/request/auto-http.ts +76 -76
  229. package/libs/request/index.ts +223 -223
  230. package/libs/store/index.ts +88 -88
  231. package/libs/util/async-validator.d.ts +62 -62
  232. package/libs/util/async-validator.js +1 -1
  233. package/libs/util/calendar.d.ts +57 -57
  234. package/libs/util/emitter.ts +102 -102
  235. package/libs/util/eventBus.ts +86 -0
  236. package/libs/util/logger.ts +364 -0
  237. package/libs/util/mitt.ts +115 -115
  238. package/libs/util/parent.ts +20 -20
  239. package/package.json +1 -1
  240. package/readme.md +241 -237
  241. package/theme.scss +38 -38
  242. package/types/components.d.ts +97 -96
  243. package/types/global.d.ts +331 -295
  244. package/types/ignore-errors.d.ts +30 -30
  245. package/types/index.d.ts +19 -19
  246. package/types/uni-app.d.ts +63 -63
@@ -0,0 +1,343 @@
1
+ // utils/useComponent.ts
2
+ import { ref, reactive, getCurrentInstance, onUnmounted, nextTick, computed } from 'vue';
3
+ import { logger } from '../util/logger';
4
+ import { eventBus } from '../util/eventBus';
5
+
6
+ // 简化类型定义
7
+ interface ParentContext {
8
+ name: string;
9
+ addChild: (child: ChildContext) => void;
10
+ removeChild: (childId: string) => void;
11
+ broadcast: (event: string, data?: any) => void;
12
+ getChildren: () => ChildContext[];
13
+ getExposed: () => Record<string, any>;
14
+ getChildExposed: (childId: string) => Record<string, any>;
15
+ getChildrenExposed: () => Array<{ id: string; name: string; exposed: Record<string, any> }>;
16
+ }
17
+
18
+ interface ChildContext {
19
+ id: string;
20
+ name: string;
21
+ emitToParent: (event: string, data?: any) => void;
22
+ getParentExposed: () => Record<string, any>;
23
+ getInstance: () => any;
24
+ getExposed: () => Record<string, any>;
25
+ }
26
+
27
+ // 全局存储
28
+ const parentMap = new Map<string, ParentContext>();
29
+ const childMap = new Map<string, ChildContext>();
30
+
31
+ // 事件常量
32
+ const PARENT_REGISTERED_EVENT = 'parent:registered';
33
+ const PARENT_UNMOUNTED_EVENT = 'parent:unmounted';
34
+ const CHILD_REGISTERED_EVENT = 'child:registered';
35
+
36
+ // 热更新清理函数
37
+ export function cleanupComponentRelations(): void {
38
+ logger.log('Cleaning up component relations for hot reload');
39
+ parentMap.clear();
40
+ childMap.clear();
41
+ }
42
+
43
+ // 热更新处理
44
+ if (import.meta.hot) {
45
+ import.meta.hot.accept(() => {
46
+ setTimeout(() => {
47
+ cleanupComponentRelations();
48
+ }, 50);
49
+ });
50
+ }
51
+
52
+ /**
53
+ * 生成实例唯一ID
54
+ */
55
+ function generateInstanceId(componentName: string): string {
56
+ return `${componentName}_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
57
+ }
58
+
59
+ /**
60
+ * 父组件 Hook
61
+ */
62
+ export function useParent(componentName: string) {
63
+ const instance = getCurrentInstance();
64
+ if (!instance) {
65
+ throw new Error('useParent must be called within setup function');
66
+ }
67
+
68
+ if (!componentName) {
69
+ throw new Error('Component name is required for useParent');
70
+ }
71
+
72
+ // 热更新时清理旧的父组件
73
+ if (parentMap.has(componentName)) {
74
+ logger.log(`Cleaning up existing parent ${componentName} for hot reload`);
75
+ parentMap.delete(componentName);
76
+ }
77
+
78
+ const children = reactive<ChildContext[]>([]);
79
+
80
+ // 父组件上下文
81
+ const parentContext: ParentContext = {
82
+ name: componentName,
83
+
84
+ addChild(child: ChildContext) {
85
+ if (!children.find(c => c.id === child.id)) {
86
+ children.push(child);
87
+ logger.log(`Parent ${componentName} added child: ${child.name}`);
88
+ }
89
+ },
90
+
91
+ removeChild(childId: string) {
92
+ const index = children.findIndex(c => c.id === childId);
93
+ if (index > -1) {
94
+ children.splice(index, 1);
95
+ logger.log(`Parent ${componentName} removed child: ${childId}`);
96
+ }
97
+ },
98
+
99
+ broadcast(event: string, data?: any) {
100
+ logger.log(`Parent ${componentName} broadcasting event: ${event}`);
101
+ children.forEach(child => {
102
+ eventBus.emit(`child:${child.id}:${event}`, data);
103
+ });
104
+ },
105
+
106
+ getChildren() {
107
+ return [...children];
108
+ },
109
+
110
+ getExposed() {
111
+ return instance.exposed || {};
112
+ },
113
+
114
+ getChildExposed(childId: string) {
115
+ const child = children.find(c => c.id === childId);
116
+ if (child && child.getExposed) {
117
+ return child.getExposed();
118
+ }
119
+ logger.warn(`Child ${childId} not found or does not have getExposed method`);
120
+ return {};
121
+ },
122
+
123
+ getChildrenExposed() {
124
+ return children
125
+ .filter(child => child.getExposed)
126
+ .map(child => {
127
+ const exposed = child.getExposed();
128
+ return {
129
+ id: child.id,
130
+ name: child.name,
131
+ exposed: exposed
132
+ };
133
+ })
134
+ .filter(item => Object.keys(item.exposed).length > 0);
135
+ }
136
+ };
137
+
138
+ // 注册父组件并广播事件
139
+ parentMap.set(componentName, parentContext);
140
+ logger.log(`Parent ${componentName} registered`);
141
+
142
+ // 广播父组件注册事件
143
+ eventBus.emit(PARENT_REGISTERED_EVENT, { name: componentName, parent: parentContext });
144
+
145
+ // 组件卸载时清理
146
+ onUnmounted(() => {
147
+ parentMap.delete(componentName);
148
+ eventBus.emit(PARENT_UNMOUNTED_EVENT, { name: componentName });
149
+ logger.log(`Parent ${componentName} unmounted`);
150
+ });
151
+
152
+ return {
153
+ parentName: componentName,
154
+ children,
155
+ broadcast: parentContext.broadcast,
156
+ getChildren: parentContext.getChildren,
157
+ getChildExposed: parentContext.getChildExposed,
158
+ getChildrenExposed: parentContext.getChildrenExposed
159
+ };
160
+ }
161
+
162
+ /**
163
+ * 子组件 Hook
164
+ */
165
+ export function useChildren(componentName: string, parentName: string) {
166
+ const instance = getCurrentInstance();
167
+ if (!instance) {
168
+ throw new Error('useChildren must be called within setup function');
169
+ }
170
+
171
+ if (!componentName || !parentName) {
172
+ throw new Error('Component name and parent name are required for useChildren');
173
+ }
174
+
175
+ const instanceId = generateInstanceId(componentName);
176
+ const parentRef = ref<ParentContext | null>(null);
177
+ const parentExposed = ref<Record<string, any>>({});
178
+
179
+ // 热更新时清理旧的子组件
180
+ if (childMap.has(instanceId)) {
181
+ logger.log(`Cleaning up existing child ${componentName} for hot reload`);
182
+ childMap.delete(instanceId);
183
+ }
184
+
185
+ // 获取父组件暴露内容
186
+ const getParentExposed = (): Record<string, any> => {
187
+ if (parentRef.value) {
188
+ const exposed = parentRef.value.getExposed();
189
+ parentExposed.value = exposed;
190
+ return exposed;
191
+ }
192
+ return {};
193
+ };
194
+
195
+ // 获取子组件exposed内容
196
+ const getExposed = (): Record<string, any> => {
197
+ return instance.exposed || {};
198
+ };
199
+
200
+ // 链接到父组件
201
+ const linkParent = (): boolean => {
202
+ const parent = parentMap.get(parentName);
203
+ if (parent) {
204
+ parentRef.value = parent;
205
+ parent.addChild(childContext);
206
+ getParentExposed();
207
+ logger.log(`Child ${componentName} linked to parent ${parentName}`);
208
+ return true;
209
+ }
210
+ return false;
211
+ };
212
+
213
+ // 子组件上下文
214
+ const childContext: ChildContext = {
215
+ id: instanceId,
216
+ name: componentName,
217
+
218
+ emitToParent(event: string, data?: any) {
219
+ eventBus.emit(`parent:${parentName}:${event}`, {
220
+ data,
221
+ childId: instanceId,
222
+ childName: componentName
223
+ });
224
+ },
225
+
226
+ getParentExposed,
227
+ getInstance() {
228
+ return instance;
229
+ },
230
+ getExposed
231
+ };
232
+
233
+ // 注册子组件
234
+ childMap.set(instanceId, childContext);
235
+ logger.log(`Child ${componentName} registered`);
236
+
237
+ // 广播子组件注册事件
238
+ eventBus.emit(CHILD_REGISTERED_EVENT, {
239
+ id: instanceId,
240
+ name: componentName,
241
+ parentName: parentName
242
+ });
243
+
244
+ // 立即尝试连接父组件
245
+ let connected = linkParent();
246
+
247
+ // 如果没连接上,监听父组件注册事件
248
+ if (!connected) {
249
+ const parentRegisteredHandler = (eventData: any) => {
250
+ if (eventData.name === parentName) {
251
+ connected = linkParent();
252
+ if (connected) {
253
+ eventBus.off(PARENT_REGISTERED_EVENT, parentRegisteredHandler);
254
+ }
255
+ }
256
+ };
257
+ eventBus.on(PARENT_REGISTERED_EVENT, parentRegisteredHandler);
258
+ }
259
+
260
+ // 监听父组件卸载事件
261
+ const parentUnmountedHandler = (eventData: any) => {
262
+ if (eventData.name === parentName && parentRef.value) {
263
+ parentRef.value = null;
264
+ parentExposed.value = {};
265
+ logger.log(`Parent ${parentName} unmounted, child ${componentName} disconnected`);
266
+ }
267
+ };
268
+ eventBus.on(PARENT_UNMOUNTED_EVENT, parentUnmountedHandler);
269
+
270
+ // 组件卸载时清理
271
+ onUnmounted(() => {
272
+ if (parentRef.value) {
273
+ parentRef.value.removeChild(instanceId);
274
+ }
275
+ childMap.delete(instanceId);
276
+ eventBus.off(PARENT_REGISTERED_EVENT);
277
+ eventBus.off(PARENT_UNMOUNTED_EVENT, parentUnmountedHandler);
278
+ logger.log(`Child ${componentName} unmounted`);
279
+ });
280
+
281
+ return {
282
+ childId: instanceId,
283
+ childName: componentName,
284
+ parent: parentRef,
285
+ emitToParent: childContext.emitToParent,
286
+ getParentExposed,
287
+ parentExposed: computed(() => parentExposed.value),
288
+ getExposed: childContext.getExposed
289
+ };
290
+ }
291
+
292
+ /**
293
+ * 监听子组件事件
294
+ */
295
+ export function onChildEvent(parentName: string, event: string, handler: Function) {
296
+ eventBus.on(`parent:${parentName}:${event}`, eventData => {
297
+ handler(eventData.data, eventData.childId, eventData.childName);
298
+ });
299
+ }
300
+
301
+ /**
302
+ * 监听父组件事件
303
+ */
304
+ export function onParentEvent(childId: string, event: string, handler: Function) {
305
+ // 修复类型问题:将 Function 转换为 EventCallback
306
+ const eventCallback = (data?: any, ...args: any[]) => {
307
+ handler(data, ...args);
308
+ };
309
+ eventBus.on(`child:${childId}:${event}`, eventCallback);
310
+
311
+ // 返回取消监听函数
312
+ return () => {
313
+ eventBus.off(`child:${childId}:${event}`, eventCallback);
314
+ };
315
+ }
316
+
317
+ /**
318
+ * 检查父组件是否存在
319
+ */
320
+ export function hasParent(parentName: string): boolean {
321
+ return parentMap.has(parentName);
322
+ }
323
+
324
+ /**
325
+ * 获取所有已注册的父组件名称
326
+ */
327
+ export function getRegisteredParents(): string[] {
328
+ return Array.from(parentMap.keys());
329
+ }
330
+
331
+ /**
332
+ * 获取父组件实例
333
+ */
334
+ export function getParent(parentName: string): ParentContext | undefined {
335
+ return parentMap.get(parentName);
336
+ }
337
+
338
+ /**
339
+ * 获取子组件实例
340
+ */
341
+ export function getChild(childId: string): ChildContext | undefined {
342
+ return childMap.get(childId);
343
+ }
@@ -1,77 +1,77 @@
1
- import { type ComponentInternalInstance, getCurrentInstance } from 'vue';
2
-
3
- /**
4
- * 将事件名转换为驼峰格式
5
- * @param str 需要转换的字符串
6
- * @description 例如:on-form-change -> onFormChange
7
- * @returns
8
- */
9
- function formatToCamelCase(str: string): string {
10
- return str.replace(/-([a-z])/g, function (g) {
11
- return g[1].toUpperCase();
12
- });
13
- }
14
-
15
- export function useEmitter(name: string) {
16
- const instance: ComponentInternalInstance | null | undefined = getCurrentInstance();
17
-
18
- /** * 向上查找父组件并派发事件
19
- * @param instance 当前组件实例(setup中可用getCurrentInstance())
20
- * @param componentName 目标组件名
21
- * @param eventName 事件名
22
- * @param params 参数
23
- */
24
- function dispatch(componentName: string, eventName: string, ...params: any[]) {
25
- let parent = instance && (instance.parent as ComponentInternalInstance | null | undefined);
26
- while (parent) {
27
- const name = (parent.type as any)?.name as string | undefined;
28
- if (name === componentName) {
29
- // 找到目标组件,派发事件
30
- // Vue3未解决,目标组件事件监听失效,待优化,暂时使用下面的方式解决,如果你有好的方式也可以告诉我或者提PR
31
- parent.emit && parent.emit(eventName, ...params);
32
- // 如果有对应的方法,执行方法
33
- // 这里可以考虑将 eventName 转换为驼峰格式
34
- // 例如:on-form-change -> onFormChange
35
- parent.exposed?.[formatToCamelCase(eventName)] && parent.exposed[formatToCamelCase(eventName)](...params);
36
- break;
37
- }
38
- parent = parent.parent;
39
- }
40
- }
41
-
42
- /**
43
- * 向下递归查找子组件并广播事件
44
- * @param instance 当前组件实例(setup中可用getCurrentInstance())
45
- * @param componentName 目标组件名
46
- * @param eventName 事件名
47
- * @param params 参数
48
- */
49
- function broadcast(componentName: string, eventName: string, ...params: any[]) {
50
- if (!instance) return;
51
- const subTree = (instance.subTree as any)?.children || [];
52
- const children = Array.isArray(subTree) ? subTree : [subTree];
53
- children.forEach((vnode: any) => {
54
- const child = vnode.component as ComponentInternalInstance | undefined;
55
-
56
- if (child) {
57
- const name = (child.type as any)?.name as string | undefined;
58
- if (name === componentName) {
59
- // 找到目标组件,广播事件
60
- // Vue3未解决,目标组件事件监听失效,待优化,暂时使用下面的方式解决,如果你有好的方式也可以告诉我或者提PR
61
- child.emit && child.emit(eventName, ...params);
62
- // 如果有对应的方法,执行方法
63
- // 这里可以考虑将 eventName 转换为驼峰格式
64
- // 例如:on-form-change -> onFormChange
65
- child.exposed?.[formatToCamelCase(eventName)] && child.exposed[formatToCamelCase(eventName)](...params);
66
- } else {
67
- broadcast.call(child, componentName, eventName, ...params);
68
- }
69
- }
70
- });
71
- }
72
-
73
- return {
74
- dispatch,
75
- broadcast
76
- };
77
- }
1
+ import { type ComponentInternalInstance, getCurrentInstance } from 'vue';
2
+
3
+ /**
4
+ * 将事件名转换为驼峰格式
5
+ * @param str 需要转换的字符串
6
+ * @description 例如:on-form-change -> onFormChange
7
+ * @returns
8
+ */
9
+ function formatToCamelCase(str: string): string {
10
+ return str.replace(/-([a-z])/g, function (g) {
11
+ return g[1].toUpperCase();
12
+ });
13
+ }
14
+
15
+ export function useEmitter(name: string) {
16
+ const instance: ComponentInternalInstance | null | undefined = getCurrentInstance();
17
+
18
+ /** * 向上查找父组件并派发事件
19
+ * @param instance 当前组件实例(setup中可用getCurrentInstance())
20
+ * @param componentName 目标组件名
21
+ * @param eventName 事件名
22
+ * @param params 参数
23
+ */
24
+ function dispatch(componentName: string, eventName: string, ...params: any[]) {
25
+ let parent = instance && (instance.parent as ComponentInternalInstance | null | undefined);
26
+ while (parent) {
27
+ const name = (parent.type as any)?.name as string | undefined;
28
+ if (name === componentName) {
29
+ // 找到目标组件,派发事件
30
+ // Vue3未解决,目标组件事件监听失效,待优化,暂时使用下面的方式解决,如果你有好的方式也可以告诉我或者提PR
31
+ parent.emit && parent.emit(eventName, ...params);
32
+ // 如果有对应的方法,执行方法
33
+ // 这里可以考虑将 eventName 转换为驼峰格式
34
+ // 例如:on-form-change -> onFormChange
35
+ parent.exposed?.[formatToCamelCase(eventName)] && parent.exposed[formatToCamelCase(eventName)](...params);
36
+ break;
37
+ }
38
+ parent = parent.parent;
39
+ }
40
+ }
41
+
42
+ /**
43
+ * 向下递归查找子组件并广播事件
44
+ * @param instance 当前组件实例(setup中可用getCurrentInstance())
45
+ * @param componentName 目标组件名
46
+ * @param eventName 事件名
47
+ * @param params 参数
48
+ */
49
+ function broadcast(componentName: string, eventName: string, ...params: any[]) {
50
+ if (!instance) return;
51
+ const subTree = (instance.subTree as any)?.children || [];
52
+ const children = Array.isArray(subTree) ? subTree : [subTree];
53
+ children.forEach((vnode: any) => {
54
+ const child = vnode.component as ComponentInternalInstance | undefined;
55
+
56
+ if (child) {
57
+ const name = (child.type as any)?.name as string | undefined;
58
+ if (name === componentName) {
59
+ // 找到目标组件,广播事件
60
+ // Vue3未解决,目标组件事件监听失效,待优化,暂时使用下面的方式解决,如果你有好的方式也可以告诉我或者提PR
61
+ child.emit && child.emit(eventName, ...params);
62
+ // 如果有对应的方法,执行方法
63
+ // 这里可以考虑将 eventName 转换为驼峰格式
64
+ // 例如:on-form-change -> onFormChange
65
+ child.exposed?.[formatToCamelCase(eventName)] && child.exposed[formatToCamelCase(eventName)](...params);
66
+ } else {
67
+ broadcast.call(child, componentName, eventName, ...params);
68
+ }
69
+ }
70
+ });
71
+ }
72
+
73
+ return {
74
+ dispatch,
75
+ broadcast
76
+ };
77
+ }
@@ -1,31 +1,33 @@
1
- import { getCurrentInstance, onUnmounted } from 'vue';
2
-
3
- export function useParent(name: string) {
4
- const instance = getCurrentInstance();
5
- if (!instance) return { parent: null };
6
-
7
- // 查找父组件
8
- let parent: any = instance.parent;
9
- while (parent) {
10
- const parentName = parent.type?.name;
11
- if (parentName === name) break;
12
- parent = parent.parent;
13
- }
14
-
15
- // 建立父子关系
16
- if (parent) {
17
- (parent as any).children = (parent as any).children || [];
18
- (parent as any).children.push(instance);
19
- // 卸载时移除
20
- onUnmounted(() => {
21
- const i = parent.children.indexOf(instance);
22
- i > -1 && parent.children.splice(i, 1);
23
- });
24
- }
25
-
26
- return {
27
- parent,
28
- parentChildren: parent?.children || [],
29
- parentExposed: parent?.exposed || null
30
- };
31
- }
1
+ import { getCurrentInstance, onUnmounted } from 'vue';
2
+
3
+ export function useParent(name: string) {
4
+ const instance = getCurrentInstance();
5
+
6
+ function getParent() {
7
+ if (!instance) return null;
8
+
9
+ // 查找父组件
10
+ let parent: any = instance.parent;
11
+ while (parent) {
12
+ const parentName = parent.type?.name;
13
+ if (parentName === name) break;
14
+ parent = parent.parent;
15
+ }
16
+
17
+ // 建立父子关系
18
+ if (parent) {
19
+ (parent as any).children = (parent as any).children || [];
20
+ (parent as any).children.push(instance);
21
+ // 卸载时移除
22
+ onUnmounted(() => {
23
+ const i = parent.children.indexOf(instance);
24
+ i > -1 && parent.children.splice(i, 1);
25
+ });
26
+ }
27
+ return parent;
28
+ }
29
+
30
+ return {
31
+ getParent
32
+ };
33
+ }
@@ -1,33 +1,33 @@
1
- import { getCurrentInstance, nextTick, ref } from 'vue';
2
-
3
- /**
4
- * useRect - 获取元素的位置信息(响应式,原生实现)
5
- * @param selector 选择器(如 #id 或 .class)
6
- * @param all 是否获取所有匹配元素
7
- * @returns rect 响应式的节点信息,refresh 主动刷新方法
8
- */
9
- export function useRect(selector: string, all = false) {
10
- const rect = ref<any>(all ? [] : null);
11
- const instance = getCurrentInstance();
12
-
13
- async function refresh(delay = 0) {
14
- await nextTick();
15
- return new Promise(resolve => {
16
- setTimeout(() => {
17
- uni.createSelectorQuery()
18
- .in(instance?.proxy)
19
- [all ? 'selectAll' : 'select'](selector)
20
- .boundingClientRect((res: any) => {
21
- rect.value = res;
22
- resolve(res);
23
- })
24
- .exec();
25
- }, delay);
26
- });
27
- }
28
-
29
- return {
30
- rect,
31
- refresh
32
- };
33
- }
1
+ import { getCurrentInstance, nextTick, ref } from 'vue';
2
+
3
+ /**
4
+ * useRect - 获取元素的位置信息(响应式,原生实现)
5
+ * @param selector 选择器(如 #id 或 .class)
6
+ * @param all 是否获取所有匹配元素
7
+ * @returns rect 响应式的节点信息,refresh 主动刷新方法
8
+ */
9
+ export function useRect(selector: string, all = false) {
10
+ const rect = ref<any>(all ? [] : null);
11
+ const instance = getCurrentInstance();
12
+
13
+ async function refreshRect(delay = 0) {
14
+ await nextTick();
15
+ return new Promise(resolve => {
16
+ setTimeout(() => {
17
+ uni.createSelectorQuery()
18
+ .in(instance?.proxy)
19
+ [all ? 'selectAll' : 'select'](selector)
20
+ .boundingClientRect((res: any) => {
21
+ rect.value = res;
22
+ resolve(res);
23
+ })
24
+ .exec();
25
+ }, delay);
26
+ });
27
+ }
28
+
29
+ return {
30
+ rect,
31
+ refreshRect
32
+ };
33
+ }