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.
- package/changelog.md +522 -481
- package/components/common/props.ts +22 -22
- package/components/u-action-sheet/types.ts +35 -35
- package/components/u-action-sheet/u-action-sheet.vue +160 -160
- package/components/u-alert-tips/types.ts +39 -39
- package/components/u-alert-tips/u-alert-tips.vue +212 -212
- package/components/u-avatar/types.ts +34 -34
- package/components/u-avatar/u-avatar.vue +193 -193
- package/components/u-avatar-cropper/types.ts +23 -23
- package/components/u-avatar-cropper/u-avatar-cropper.vue +286 -286
- package/components/u-avatar-cropper/weCropper.d.ts +62 -62
- package/components/u-avatar-cropper/weCropper.js +1253 -1253
- package/components/u-avatar-cropper/weCropper.ts +1255 -1255
- package/components/u-back-top/types.ts +39 -39
- package/components/u-back-top/u-back-top.vue +125 -125
- package/components/u-badge/types.ts +36 -36
- package/components/u-badge/u-badge.vue +165 -165
- package/components/u-button/types.ts +66 -66
- package/components/u-button/u-button.vue +556 -556
- package/components/u-calendar/types.ts +73 -73
- package/components/u-calendar/u-calendar.vue +638 -638
- package/components/u-car-keyboard/types.ts +12 -12
- package/components/u-car-keyboard/u-car-keyboard.vue +234 -234
- package/components/u-card/types.ts +59 -59
- package/components/u-card/u-card.vue +194 -194
- package/components/u-cell-group/types.ts +17 -17
- package/components/u-cell-group/u-cell-group.vue +50 -50
- package/components/u-cell-item/types.ts +54 -54
- package/components/u-cell-item/u-cell-item.vue +202 -202
- package/components/u-checkbox/types.ts +31 -31
- package/components/u-checkbox/u-checkbox.vue +286 -267
- package/components/u-checkbox-group/types.ts +32 -32
- package/components/u-checkbox-group/u-checkbox-group.vue +130 -79
- package/components/u-circle-progress/types.ts +52 -52
- package/components/u-circle-progress/u-circle-progress.vue +187 -187
- package/components/u-city-select/types.ts +20 -20
- package/components/u-city-select/u-city-select.vue +236 -236
- package/components/u-col/types.ts +30 -30
- package/components/u-col/u-col.vue +123 -123
- package/components/u-collapse/types.ts +33 -33
- package/components/u-collapse/u-collapse.vue +195 -69
- package/components/u-collapse-item/types.ts +27 -27
- package/components/u-collapse-item/u-collapse-item.vue +311 -201
- package/components/u-column-notice/types.ts +48 -48
- package/components/u-column-notice/u-column-notice.vue +176 -176
- package/components/u-count-down/types.ts +42 -42
- package/components/u-count-down/u-count-down.vue +258 -258
- package/components/u-count-to/types.ts +32 -32
- package/components/u-count-to/u-count-to.vue +241 -241
- package/components/u-divider/types.ts +31 -31
- package/components/u-divider/u-divider.vue +121 -121
- package/components/u-dropdown/types.ts +32 -32
- package/components/u-dropdown/u-dropdown.vue +289 -289
- package/components/u-dropdown-item/types.ts +27 -27
- package/components/u-dropdown-item/u-dropdown-item.vue +123 -123
- package/components/u-empty/types.ts +36 -36
- package/components/u-empty/u-empty.vue +88 -88
- package/components/u-field/types.ts +69 -69
- package/components/u-field/u-field.vue +354 -354
- package/components/u-form/u-form.vue +132 -132
- package/components/u-form-item/u-form-item.vue +417 -417
- package/components/u-full-screen/types.ts +14 -14
- package/components/u-full-screen/u-full-screen.vue +82 -82
- package/components/u-gap/types.ts +18 -18
- package/components/u-gap/u-gap.vue +40 -40
- package/components/u-grid/types.ts +19 -19
- package/components/u-grid/u-grid.vue +93 -93
- package/components/u-grid-item/types.ts +16 -16
- package/components/u-grid-item/u-grid-item.vue +130 -130
- package/components/u-icon/types.ts +62 -62
- package/components/u-icon/u-icon.vue +281 -281
- package/components/u-image/types.ts +51 -51
- package/components/u-image/u-image.vue +222 -222
- package/components/u-index-anchor/types.ts +16 -16
- package/components/u-index-anchor/u-index-anchor.vue +86 -86
- package/components/u-index-list/types.ts +43 -43
- package/components/u-index-list/u-index-list.vue +355 -355
- package/components/u-input/types.ts +140 -140
- package/components/u-input/u-input.vue +264 -264
- package/components/u-keyboard/types.ts +40 -40
- package/components/u-keyboard/u-keyboard.vue +158 -158
- package/components/u-lazy-load/types.ts +37 -37
- package/components/u-lazy-load/u-lazy-load.vue +233 -233
- package/components/u-line/types.ts +44 -44
- package/components/u-line/u-line.vue +59 -59
- package/components/u-line-progress/types.ts +58 -58
- package/components/u-line-progress/u-line-progress.vue +109 -109
- package/components/u-link/types.ts +43 -43
- package/components/u-link/u-link.vue +75 -75
- package/components/u-loading/types.ts +35 -35
- package/components/u-loading/u-loading.vue +90 -90
- package/components/u-loading-popup/types.ts +26 -26
- package/components/u-loading-popup/u-loading-popup.vue +239 -239
- package/components/u-loadmore/types.ts +79 -79
- package/components/u-loadmore/u-loadmore.vue +140 -140
- package/components/u-mask/types.ts +43 -43
- package/components/u-mask/u-mask.vue +106 -106
- package/components/u-message-input/types.ts +74 -74
- package/components/u-message-input/u-message-input.vue +255 -255
- package/components/u-modal/types.ts +118 -118
- package/components/u-modal/u-modal.vue +204 -204
- package/components/u-navbar/types.ts +103 -103
- package/components/u-navbar/u-navbar.vue +226 -226
- package/components/u-no-network/image.ts +2 -2
- package/components/u-no-network/types.ts +28 -28
- package/components/u-no-network/u-no-network.vue +290 -290
- package/components/u-notice-bar/types.ts +111 -111
- package/components/u-notice-bar/u-notice-bar.vue +174 -174
- package/components/u-number-box/types.ts +42 -42
- package/components/u-number-box/u-number-box.vue +312 -312
- package/components/u-number-keyboard/types.ts +26 -26
- package/components/u-number-keyboard/u-number-keyboard.vue +166 -166
- package/components/u-picker/types.ts +123 -123
- package/components/u-picker/u-picker.vue +637 -637
- package/components/u-popup/types.ts +59 -59
- package/components/u-popup/u-popup.vue +359 -359
- package/components/u-radio/types.ts +25 -25
- package/components/u-radio/u-radio.vue +258 -258
- package/components/u-radio-group/types.ts +29 -29
- package/components/u-radio-group/u-radio-group.vue +98 -98
- package/components/u-rate/types.ts +40 -40
- package/components/u-rate/u-rate.vue +234 -234
- package/components/u-read-more/types.ts +35 -35
- package/components/u-read-more/u-read-more.vue +150 -150
- package/components/u-root-portal/u-root-portal.vue +54 -0
- package/components/u-row/types.ts +20 -20
- package/components/u-row/u-row.vue +87 -87
- package/components/u-row-notice/types.ts +39 -39
- package/components/u-row-notice/u-row-notice.vue +213 -213
- package/components/u-safe-bottom/u-safe-bottom.vue +46 -46
- package/components/u-search/types.ts +53 -53
- package/components/u-search/u-search.vue +256 -256
- package/components/u-section/types.ts +32 -32
- package/components/u-section/u-section.vue +125 -125
- package/components/u-select/types.ts +43 -43
- package/components/u-select/u-select.vue +361 -361
- package/components/u-skeleton/types.ts +20 -20
- package/components/u-skeleton/u-skeleton.vue +205 -205
- package/components/u-slider/types.ts +32 -32
- package/components/u-slider/u-slider.vue +238 -238
- package/components/u-status-bar/u-status-bar.vue +65 -65
- package/components/u-steps/types.ts +28 -28
- package/components/u-steps/u-steps.vue +160 -160
- package/components/u-sticky/types.ts +22 -22
- package/components/u-sticky/u-sticky.vue +159 -159
- package/components/u-subsection/types.ts +36 -36
- package/components/u-subsection/u-subsection.vue +328 -328
- package/components/u-swipe-action/types.ts +50 -50
- package/components/u-swipe-action/u-swipe-action.vue +253 -253
- package/components/u-swiper/types.ts +47 -47
- package/components/u-swiper/u-swiper.vue +266 -266
- package/components/u-switch/types.ts +28 -28
- package/components/u-switch/u-switch.vue +136 -136
- package/components/u-tabbar/types.ts +36 -36
- package/components/u-tabbar/u-tabbar.vue +280 -280
- package/components/u-table/types.ts +25 -25
- package/components/u-table/u-table.vue +55 -55
- package/components/u-tabs/types.ts +51 -51
- package/components/u-tabs/u-tabs.vue +284 -284
- package/components/u-tabs-swiper/types.ts +53 -53
- package/components/u-tabs-swiper/u-tabs-swiper.vue +379 -379
- package/components/u-tag/types.ts +37 -37
- package/components/u-tag/u-tag.vue +244 -244
- package/components/u-td/types.ts +12 -12
- package/components/u-td/u-td.vue +87 -87
- package/components/u-text/types.ts +69 -69
- package/components/u-text/u-text.vue +326 -326
- package/components/u-th/types.ts +12 -12
- package/components/u-th/u-th.vue +81 -81
- package/components/u-time-line/u-time-line.vue +39 -39
- package/components/u-time-line-item/types.ts +14 -14
- package/components/u-time-line-item/u-time-line-item.vue +78 -78
- package/components/u-toast/types.ts +36 -36
- package/components/u-toast/u-toast.vue +233 -233
- package/components/u-top-tips/types.ts +14 -14
- package/components/u-top-tips/u-top-tips.vue +113 -113
- package/components/u-tr/types.ts +8 -8
- package/components/u-tr/u-tr.vue +24 -24
- package/components/u-upload/types.ts +74 -74
- package/components/u-upload/u-upload.vue +545 -545
- package/components/u-verification-code/types.ts +22 -22
- package/components/u-verification-code/u-verification-code.vue +164 -164
- package/components/u-waterfall/types.ts +16 -16
- package/components/u-waterfall/u-waterfall.vue +175 -175
- package/iconfont.css +912 -912
- package/index.scss +25 -25
- package/index.ts +38 -29
- package/libs/config/config.ts +26 -26
- package/libs/config/zIndex.ts +37 -37
- package/libs/css/color.scss +155 -155
- package/libs/css/common.scss +178 -178
- package/libs/css/style.components.scss +16 -16
- package/libs/css/style.h5.scss +8 -8
- package/libs/css/style.mp.scss +72 -72
- package/libs/css/style.nvue.scss +15 -15
- package/libs/css/style.vue.scss +188 -188
- package/libs/function/$parent.ts +21 -21
- package/libs/function/addUnit.ts +13 -13
- package/libs/function/color.ts +37 -37
- package/libs/function/colorGradient.ts +125 -125
- package/libs/function/debounce.ts +28 -28
- package/libs/function/deepClone.ts +39 -39
- package/libs/function/deepMerge.ts +34 -34
- package/libs/function/getParent.ts +59 -59
- package/libs/function/getRect.ts +26 -26
- package/libs/function/guid.ts +42 -42
- package/libs/function/md5.ts +391 -391
- package/libs/function/parent.ts +21 -21
- package/libs/function/queryParams.ts +60 -60
- package/libs/function/random.ts +16 -16
- package/libs/function/randomArray.ts +11 -11
- package/libs/function/route.ts +118 -118
- package/libs/function/styleUtils.ts +83 -83
- package/libs/function/sys.ts +15 -15
- package/libs/function/test.ts +285 -285
- package/libs/function/throttle.ts +31 -31
- package/libs/function/timeFormat.ts +54 -54
- package/libs/function/timeFrom.ts +48 -48
- package/libs/function/toast.ts +14 -14
- package/libs/function/trim.ts +21 -21
- package/libs/function/type2icon.ts +36 -36
- package/libs/hooks/index.ts +3 -3
- package/libs/hooks/useComponent.ts +343 -0
- package/libs/hooks/useEmitter.ts +77 -77
- package/libs/hooks/useParent.ts +33 -31
- package/libs/hooks/useRect.ts +33 -33
- package/libs/index.ts +320 -291
- package/libs/request/auto-http.ts +76 -76
- package/libs/request/index.ts +223 -223
- package/libs/store/index.ts +88 -88
- package/libs/util/async-validator.d.ts +62 -62
- package/libs/util/async-validator.js +1 -1
- package/libs/util/calendar.d.ts +57 -57
- package/libs/util/emitter.ts +102 -102
- package/libs/util/eventBus.ts +86 -0
- package/libs/util/logger.ts +364 -0
- package/libs/util/mitt.ts +115 -115
- package/libs/util/parent.ts +20 -20
- package/package.json +1 -1
- package/readme.md +241 -237
- package/theme.scss +38 -38
- package/types/components.d.ts +97 -96
- package/types/global.d.ts +331 -295
- package/types/ignore-errors.d.ts +30 -30
- package/types/index.d.ts +19 -19
- 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
|
+
}
|
package/libs/hooks/useEmitter.ts
CHANGED
|
@@ -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
|
+
}
|
package/libs/hooks/useParent.ts
CHANGED
|
@@ -1,31 +1,33 @@
|
|
|
1
|
-
import { getCurrentInstance, onUnmounted } from 'vue';
|
|
2
|
-
|
|
3
|
-
export function useParent(name: string) {
|
|
4
|
-
const instance = getCurrentInstance();
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
(parent
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
parent
|
|
28
|
-
|
|
29
|
-
|
|
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
|
+
}
|
package/libs/hooks/useRect.ts
CHANGED
|
@@ -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
|
|
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
|
-
|
|
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
|
+
}
|