raise-common-lib 0.0.216 → 0.0.218
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/bundles/raise-common-lib.umd.js +1173 -2
- package/bundles/raise-common-lib.umd.js.map +1 -1
- package/bundles/raise-common-lib.umd.min.js +1 -1
- package/bundles/raise-common-lib.umd.min.js.map +1 -1
- package/esm2015/lib/common-grid/index.component.js +5 -2
- package/esm2015/lib/dialog/common-dialog/index.component.js +6 -2
- package/esm2015/lib/raise-common-lib.module.js +4 -1
- package/esm2015/lib/smart-popup/index.component.js +1059 -0
- package/esm2015/public-api.js +2 -1
- package/esm2015/raise-common-lib.js +2 -2
- package/esm5/lib/common-grid/index.component.js +5 -2
- package/esm5/lib/dialog/common-dialog/index.component.js +6 -2
- package/esm5/lib/raise-common-lib.module.js +4 -1
- package/esm5/lib/smart-popup/index.component.js +1164 -0
- package/esm5/public-api.js +2 -1
- package/esm5/raise-common-lib.js +2 -2
- package/fesm2015/raise-common-lib.js +1070 -3
- package/fesm2015/raise-common-lib.js.map +1 -1
- package/fesm5/raise-common-lib.js +1173 -3
- package/fesm5/raise-common-lib.js.map +1 -1
- package/lib/common-grid/index.component.d.ts +1 -0
- package/lib/dialog/common-dialog/index.component.d.ts +1 -0
- package/lib/smart-popup/index.component.d.ts +55 -0
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
- package/raise-common-lib.metadata.json +1 -1
|
@@ -0,0 +1,1059 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview added by tsickle
|
|
3
|
+
* Generated from: lib/smart-popup/index.component.ts
|
|
4
|
+
* @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
|
|
5
|
+
*/
|
|
6
|
+
import { ChangeDetectorRef, Component, ElementRef, EventEmitter, Input, Output, ViewChild, HostListener, } from "@angular/core";
|
|
7
|
+
export class SmartPopupComponent {
|
|
8
|
+
/**
|
|
9
|
+
* @param {?} ref
|
|
10
|
+
*/
|
|
11
|
+
constructor(ref) {
|
|
12
|
+
this.ref = ref;
|
|
13
|
+
this._placement = "BottomLeft";
|
|
14
|
+
this._offset = 8;
|
|
15
|
+
this._autoAdjust = "yes";
|
|
16
|
+
this.triggerElementRef = null;
|
|
17
|
+
this.loading = false;
|
|
18
|
+
this._open = false;
|
|
19
|
+
this._width = null;
|
|
20
|
+
this._height = null;
|
|
21
|
+
this.openChange = new EventEmitter();
|
|
22
|
+
this.contentLoad = new EventEmitter();
|
|
23
|
+
this.currentTriggerElement = null;
|
|
24
|
+
this.FIXED_CONTAINER_ID = "rs-smart-popup-fixed-container";
|
|
25
|
+
this.opened = "no";
|
|
26
|
+
this.positioning = "no";
|
|
27
|
+
this.actualPlacement = "bottom";
|
|
28
|
+
this.actualPosition = "start";
|
|
29
|
+
this.onClickOutside = (/**
|
|
30
|
+
* @param {?} event
|
|
31
|
+
* @return {?}
|
|
32
|
+
*/
|
|
33
|
+
(event) => {
|
|
34
|
+
/** @type {?} */
|
|
35
|
+
const target = (/** @type {?} */ (event.target));
|
|
36
|
+
/** @type {?} */
|
|
37
|
+
const isClickOnTrigger = this.currentTriggerElement && this.currentTriggerElement.contains(target);
|
|
38
|
+
/** @type {?} */
|
|
39
|
+
const isClickOnPopup = this.popupElement &&
|
|
40
|
+
this.popupElement.nativeElement &&
|
|
41
|
+
this.popupElement.nativeElement.contains(target);
|
|
42
|
+
if (!isClickOnTrigger && !isClickOnPopup) {
|
|
43
|
+
this.close();
|
|
44
|
+
}
|
|
45
|
+
});
|
|
46
|
+
this.onWindowScroll = (/**
|
|
47
|
+
* @return {?}
|
|
48
|
+
*/
|
|
49
|
+
() => {
|
|
50
|
+
if (this.opened === "yes") {
|
|
51
|
+
this.updatePopupPosition();
|
|
52
|
+
}
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
// 解析placement格式
|
|
56
|
+
/**
|
|
57
|
+
* @return {?}
|
|
58
|
+
*/
|
|
59
|
+
get placementInfo() {
|
|
60
|
+
/** @type {?} */
|
|
61
|
+
const placementMap = {
|
|
62
|
+
TopLeft: { placement: "top", position: "start" },
|
|
63
|
+
TopCenter: { placement: "top", position: "center" },
|
|
64
|
+
TopRight: { placement: "top", position: "end" },
|
|
65
|
+
BottomLeft: { placement: "bottom", position: "start" },
|
|
66
|
+
BottomCenter: { placement: "bottom", position: "center" },
|
|
67
|
+
BottomRight: { placement: "bottom", position: "end" },
|
|
68
|
+
LeftTop: { placement: "left", position: "start" },
|
|
69
|
+
LeftCenter: { placement: "left", position: "center" },
|
|
70
|
+
LeftBottom: { placement: "left", position: "end" },
|
|
71
|
+
RightTop: { placement: "right", position: "start" },
|
|
72
|
+
RightCenter: { placement: "right", position: "center" },
|
|
73
|
+
RightBottom: { placement: "right", position: "end" },
|
|
74
|
+
};
|
|
75
|
+
return (placementMap[this._placement] || {
|
|
76
|
+
placement: "bottom",
|
|
77
|
+
position: "start",
|
|
78
|
+
});
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* @return {?}
|
|
82
|
+
*/
|
|
83
|
+
get offset() {
|
|
84
|
+
return this._offset || 8;
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* @return {?}
|
|
88
|
+
*/
|
|
89
|
+
get autoAdjust() {
|
|
90
|
+
return this._autoAdjust === "yes";
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* @return {?}
|
|
94
|
+
*/
|
|
95
|
+
ngAfterViewInit() {
|
|
96
|
+
this.fixedContainerEl = this.getFixedContainer();
|
|
97
|
+
this.fixedContainerEl.append(this.popupElement.nativeElement);
|
|
98
|
+
this.updateTriggerElement();
|
|
99
|
+
// 如果提供了固定尺寸,立即应用
|
|
100
|
+
if (this._width !== null &&
|
|
101
|
+
this._height !== null &&
|
|
102
|
+
this.popupElement &&
|
|
103
|
+
this.popupElement.nativeElement) {
|
|
104
|
+
/** @type {?} */
|
|
105
|
+
const popupEl = this.popupElement.nativeElement;
|
|
106
|
+
popupEl.style.width = this._width + "px";
|
|
107
|
+
popupEl.style.height = this._height + "px";
|
|
108
|
+
popupEl.style.minWidth = this._width + "px";
|
|
109
|
+
popupEl.style.maxWidth = this._width + "px";
|
|
110
|
+
popupEl.style.minHeight = this._height + "px";
|
|
111
|
+
popupEl.style.maxHeight = this._height + "px";
|
|
112
|
+
}
|
|
113
|
+
// 如果外部传入 open 属性为 true,自动打开
|
|
114
|
+
if (this._open) {
|
|
115
|
+
setTimeout((/**
|
|
116
|
+
* @return {?}
|
|
117
|
+
*/
|
|
118
|
+
() => this.open()), 0);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* @param {?} changes
|
|
123
|
+
* @return {?}
|
|
124
|
+
*/
|
|
125
|
+
ngOnChanges(changes) {
|
|
126
|
+
// 处理固定尺寸变化
|
|
127
|
+
if ((changes._width || changes._height) &&
|
|
128
|
+
this.popupElement &&
|
|
129
|
+
this.popupElement.nativeElement) {
|
|
130
|
+
/** @type {?} */
|
|
131
|
+
const popupEl = this.popupElement.nativeElement;
|
|
132
|
+
if (this._width !== null) {
|
|
133
|
+
popupEl.style.width = this._width + "px";
|
|
134
|
+
popupEl.style.minWidth = this._width + "px";
|
|
135
|
+
popupEl.style.maxWidth = this._width + "px";
|
|
136
|
+
}
|
|
137
|
+
if (this._height !== null) {
|
|
138
|
+
popupEl.style.height = this._height + "px";
|
|
139
|
+
popupEl.style.minHeight = this._height + "px";
|
|
140
|
+
popupEl.style.maxHeight = this._height + "px";
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
// 先处理触发元素变化(确保在打开前更新触发元素)
|
|
144
|
+
if (changes.triggerElementRef) {
|
|
145
|
+
/** @type {?} */
|
|
146
|
+
const previousTrigger = this.currentTriggerElement;
|
|
147
|
+
this.updateTriggerElement();
|
|
148
|
+
// 如果弹窗已经打开,且触发元素发生了变化,立即更新位置
|
|
149
|
+
if (this.opened === "yes" &&
|
|
150
|
+
previousTrigger !== this.currentTriggerElement &&
|
|
151
|
+
this.currentTriggerElement) {
|
|
152
|
+
// 使用新的打开方式更新位置
|
|
153
|
+
this.positioning = "yes";
|
|
154
|
+
this.ref.markForCheck();
|
|
155
|
+
requestAnimationFrame((/**
|
|
156
|
+
* @return {?}
|
|
157
|
+
*/
|
|
158
|
+
() => {
|
|
159
|
+
requestAnimationFrame((/**
|
|
160
|
+
* @return {?}
|
|
161
|
+
*/
|
|
162
|
+
() => {
|
|
163
|
+
this.updatePopupPosition();
|
|
164
|
+
this.positioning = "no";
|
|
165
|
+
this.ref.markForCheck();
|
|
166
|
+
}));
|
|
167
|
+
}));
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
// 处理 open 变化(确保在触发元素更新后处理)
|
|
171
|
+
if (changes._open) {
|
|
172
|
+
if (this._open && this.opened === "no") {
|
|
173
|
+
// 确保触发元素已更新
|
|
174
|
+
this.updateTriggerElement();
|
|
175
|
+
// 使用 setTimeout 确保所有变更都已处理
|
|
176
|
+
setTimeout((/**
|
|
177
|
+
* @return {?}
|
|
178
|
+
*/
|
|
179
|
+
() => {
|
|
180
|
+
// 再次确保触发元素已更新
|
|
181
|
+
this.updateTriggerElement();
|
|
182
|
+
this.open();
|
|
183
|
+
}), 0);
|
|
184
|
+
}
|
|
185
|
+
else if (!this._open && this.opened === "yes") {
|
|
186
|
+
this.close();
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
// 当加载完成时,更新位置(如果提供了固定尺寸,不需要更新位置)
|
|
190
|
+
if (changes.loading &&
|
|
191
|
+
!changes.loading.currentValue &&
|
|
192
|
+
changes.loading.previousValue) {
|
|
193
|
+
if (this.opened === "yes") {
|
|
194
|
+
// 如果提供了固定尺寸,位置已经计算好了,不需要更新
|
|
195
|
+
if (this._width !== null && this._height !== null) {
|
|
196
|
+
this.contentLoad.emit();
|
|
197
|
+
}
|
|
198
|
+
else {
|
|
199
|
+
// 如果正在定位,不需要更新位置
|
|
200
|
+
if (this.positioning === "no") {
|
|
201
|
+
requestAnimationFrame((/**
|
|
202
|
+
* @return {?}
|
|
203
|
+
*/
|
|
204
|
+
() => {
|
|
205
|
+
this.updatePopupPosition();
|
|
206
|
+
this.contentLoad.emit();
|
|
207
|
+
}));
|
|
208
|
+
}
|
|
209
|
+
else {
|
|
210
|
+
// 如果正在定位,等待定位完成后再触发事件
|
|
211
|
+
/** @type {?} */
|
|
212
|
+
const checkPositioning = (/**
|
|
213
|
+
* @return {?}
|
|
214
|
+
*/
|
|
215
|
+
() => {
|
|
216
|
+
if (this.positioning === "no") {
|
|
217
|
+
this.contentLoad.emit();
|
|
218
|
+
}
|
|
219
|
+
else {
|
|
220
|
+
setTimeout(checkPositioning, 10);
|
|
221
|
+
}
|
|
222
|
+
});
|
|
223
|
+
checkPositioning();
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* @return {?}
|
|
231
|
+
*/
|
|
232
|
+
ngOnDestroy() {
|
|
233
|
+
if (this.popupElement && this.popupElement.nativeElement) {
|
|
234
|
+
this.popupElement.nativeElement.remove();
|
|
235
|
+
}
|
|
236
|
+
window.removeEventListener("click", this.onClickOutside);
|
|
237
|
+
window.removeEventListener("scroll", this.onWindowScroll, true);
|
|
238
|
+
}
|
|
239
|
+
/**
|
|
240
|
+
* @return {?}
|
|
241
|
+
*/
|
|
242
|
+
updateTriggerElement() {
|
|
243
|
+
if (this.triggerElementRef) {
|
|
244
|
+
this.currentTriggerElement = this.triggerElementRef;
|
|
245
|
+
}
|
|
246
|
+
else if (this.internalTriggerElement &&
|
|
247
|
+
this.internalTriggerElement.nativeElement) {
|
|
248
|
+
this.currentTriggerElement = this.internalTriggerElement.nativeElement;
|
|
249
|
+
}
|
|
250
|
+
else {
|
|
251
|
+
this.currentTriggerElement = null;
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
/**
|
|
255
|
+
* @return {?}
|
|
256
|
+
*/
|
|
257
|
+
getFixedContainer() {
|
|
258
|
+
/** @type {?} */
|
|
259
|
+
let containerElement = document.getElementById(this.FIXED_CONTAINER_ID);
|
|
260
|
+
if (!containerElement) {
|
|
261
|
+
containerElement = document.createElement("div");
|
|
262
|
+
containerElement.setAttribute("id", this.FIXED_CONTAINER_ID);
|
|
263
|
+
document.body.append(containerElement);
|
|
264
|
+
}
|
|
265
|
+
return containerElement;
|
|
266
|
+
}
|
|
267
|
+
/**
|
|
268
|
+
* @param {?=} triggerElement
|
|
269
|
+
* @return {?}
|
|
270
|
+
*/
|
|
271
|
+
open(triggerElement) {
|
|
272
|
+
// 如果传入了触发元素,使用它
|
|
273
|
+
if (triggerElement) {
|
|
274
|
+
this.currentTriggerElement = triggerElement;
|
|
275
|
+
}
|
|
276
|
+
else {
|
|
277
|
+
this.updateTriggerElement();
|
|
278
|
+
}
|
|
279
|
+
if (!this.currentTriggerElement) {
|
|
280
|
+
console.warn("SmartPopupComponent: No trigger element found");
|
|
281
|
+
return;
|
|
282
|
+
}
|
|
283
|
+
// 如果已经打开,先关闭再打开(更新位置)
|
|
284
|
+
if (this.opened === "yes") {
|
|
285
|
+
// 先关闭弹窗
|
|
286
|
+
this.opened = "no";
|
|
287
|
+
this.positioning = "no";
|
|
288
|
+
this.ref.markForCheck();
|
|
289
|
+
// 立即在新位置打开
|
|
290
|
+
requestAnimationFrame((/**
|
|
291
|
+
* @return {?}
|
|
292
|
+
*/
|
|
293
|
+
() => {
|
|
294
|
+
// 再次确保触发元素已更新
|
|
295
|
+
this.updateTriggerElement();
|
|
296
|
+
this.openPopup();
|
|
297
|
+
}));
|
|
298
|
+
return;
|
|
299
|
+
}
|
|
300
|
+
// 确保触发元素已更新
|
|
301
|
+
this.updateTriggerElement();
|
|
302
|
+
this.openPopup();
|
|
303
|
+
}
|
|
304
|
+
/**
|
|
305
|
+
* @private
|
|
306
|
+
* @return {?}
|
|
307
|
+
*/
|
|
308
|
+
openPopup() {
|
|
309
|
+
// 再次确保触发元素已更新
|
|
310
|
+
this.updateTriggerElement();
|
|
311
|
+
if (!this.currentTriggerElement) {
|
|
312
|
+
console.warn("SmartPopupComponent: No trigger element found when opening popup");
|
|
313
|
+
return;
|
|
314
|
+
}
|
|
315
|
+
// 如果提供了固定尺寸,可以在打开前就计算好位置
|
|
316
|
+
if (this._width !== null && this._height !== null) {
|
|
317
|
+
// 先计算位置(不依赖DOM渲染,直接根据triggerElement计算)
|
|
318
|
+
/** @type {?} */
|
|
319
|
+
const triggerRect = this.currentTriggerElement.getBoundingClientRect();
|
|
320
|
+
/** @type {?} */
|
|
321
|
+
const viewportWidth = window.innerWidth;
|
|
322
|
+
/** @type {?} */
|
|
323
|
+
const viewportHeight = window.innerHeight;
|
|
324
|
+
/** @type {?} */
|
|
325
|
+
const popupSize = {
|
|
326
|
+
width: this._width,
|
|
327
|
+
height: this._height,
|
|
328
|
+
};
|
|
329
|
+
/** @type {?} */
|
|
330
|
+
const popupRect = (/** @type {?} */ ({
|
|
331
|
+
width: popupSize.width,
|
|
332
|
+
height: popupSize.height,
|
|
333
|
+
top: 0,
|
|
334
|
+
left: 0,
|
|
335
|
+
right: popupSize.width,
|
|
336
|
+
bottom: popupSize.height,
|
|
337
|
+
x: 0,
|
|
338
|
+
y: 0,
|
|
339
|
+
toJSON: (/**
|
|
340
|
+
* @return {?}
|
|
341
|
+
*/
|
|
342
|
+
() => ({})),
|
|
343
|
+
}));
|
|
344
|
+
/** @type {?} */
|
|
345
|
+
const targetInfo = this.placementInfo;
|
|
346
|
+
/** @type {?} */
|
|
347
|
+
let finalPlacement = targetInfo.placement;
|
|
348
|
+
/** @type {?} */
|
|
349
|
+
let finalPosition = targetInfo.position;
|
|
350
|
+
if (this.autoAdjust) {
|
|
351
|
+
/** @type {?} */
|
|
352
|
+
const optimal = this.calculateOptimalPlacement(triggerRect, popupRect, viewportWidth, viewportHeight);
|
|
353
|
+
finalPlacement = optimal.placement;
|
|
354
|
+
finalPosition = optimal.position;
|
|
355
|
+
}
|
|
356
|
+
this.actualPlacement = finalPlacement;
|
|
357
|
+
this.actualPosition = finalPosition;
|
|
358
|
+
// 计算位置坐标(不依赖DOM,直接计算)
|
|
359
|
+
/** @type {?} */
|
|
360
|
+
const position = this.calculatePositionCoordinates(triggerRect, finalPlacement, finalPosition, popupSize);
|
|
361
|
+
// 先设置打开状态,但保持隐藏(positioning = yes)
|
|
362
|
+
this.opened = "yes";
|
|
363
|
+
this.positioning = "yes";
|
|
364
|
+
this.openChange.emit(true);
|
|
365
|
+
this.ref.markForCheck();
|
|
366
|
+
// 强制更新视图,确保DOM已更新
|
|
367
|
+
this.ref.detectChanges();
|
|
368
|
+
// 如果元素已经存在,立即设置位置(不等待 requestAnimationFrame)
|
|
369
|
+
if (this.popupElement && this.popupElement.nativeElement) {
|
|
370
|
+
/** @type {?} */
|
|
371
|
+
const popupEl = this.popupElement.nativeElement;
|
|
372
|
+
// 立即设置尺寸和位置
|
|
373
|
+
popupEl.style.width = this._width + "px";
|
|
374
|
+
popupEl.style.height = this._height + "px";
|
|
375
|
+
popupEl.style.minWidth = this._width + "px";
|
|
376
|
+
popupEl.style.maxWidth = this._width + "px";
|
|
377
|
+
popupEl.style.minHeight = this._height + "px";
|
|
378
|
+
popupEl.style.maxHeight = this._height + "px";
|
|
379
|
+
popupEl.style.top = position.top;
|
|
380
|
+
popupEl.style.left = position.left;
|
|
381
|
+
popupEl.style.transform = `translate(${position.translateX}, ${position.translateY})`;
|
|
382
|
+
// 强制同步应用样式
|
|
383
|
+
popupEl.offsetHeight;
|
|
384
|
+
// 位置设置完成后,立即显示弹窗
|
|
385
|
+
this.positioning = "no";
|
|
386
|
+
this.ref.markForCheck();
|
|
387
|
+
}
|
|
388
|
+
else {
|
|
389
|
+
// 如果元素还不存在,等待渲染
|
|
390
|
+
requestAnimationFrame((/**
|
|
391
|
+
* @return {?}
|
|
392
|
+
*/
|
|
393
|
+
() => {
|
|
394
|
+
if (this.popupElement && this.popupElement.nativeElement) {
|
|
395
|
+
/** @type {?} */
|
|
396
|
+
const popupEl = this.popupElement.nativeElement;
|
|
397
|
+
// 立即设置尺寸和位置
|
|
398
|
+
popupEl.style.width = this._width + "px";
|
|
399
|
+
popupEl.style.height = this._height + "px";
|
|
400
|
+
popupEl.style.minWidth = this._width + "px";
|
|
401
|
+
popupEl.style.maxWidth = this._width + "px";
|
|
402
|
+
popupEl.style.minHeight = this._height + "px";
|
|
403
|
+
popupEl.style.maxHeight = this._height + "px";
|
|
404
|
+
popupEl.style.top = position.top;
|
|
405
|
+
popupEl.style.left = position.left;
|
|
406
|
+
popupEl.style.transform = `translate(${position.translateX}, ${position.translateY})`;
|
|
407
|
+
// 强制同步应用样式
|
|
408
|
+
popupEl.offsetHeight;
|
|
409
|
+
// 位置设置完成后,立即显示弹窗
|
|
410
|
+
this.positioning = "no";
|
|
411
|
+
this.ref.markForCheck();
|
|
412
|
+
}
|
|
413
|
+
}));
|
|
414
|
+
}
|
|
415
|
+
}
|
|
416
|
+
else {
|
|
417
|
+
// 没有固定尺寸,需要等待DOM渲染
|
|
418
|
+
this.positioning = "yes";
|
|
419
|
+
this.ref.markForCheck();
|
|
420
|
+
// 立即强制更新视图,确保DOM已更新
|
|
421
|
+
this.ref.detectChanges();
|
|
422
|
+
// 使用 requestAnimationFrame 确保在浏览器渲染之前更新位置
|
|
423
|
+
requestAnimationFrame((/**
|
|
424
|
+
* @return {?}
|
|
425
|
+
*/
|
|
426
|
+
() => {
|
|
427
|
+
// 确保元素已渲染并获取尺寸
|
|
428
|
+
if (this.popupElement && this.popupElement.nativeElement) {
|
|
429
|
+
// 强制重新计算布局,确保元素尺寸已计算
|
|
430
|
+
this.popupElement.nativeElement.offsetHeight;
|
|
431
|
+
// 计算并设置位置
|
|
432
|
+
this.updatePopupPosition();
|
|
433
|
+
// 位置设置完成后,移除定位状态,显示弹窗
|
|
434
|
+
this.positioning = "no";
|
|
435
|
+
this.ref.markForCheck();
|
|
436
|
+
}
|
|
437
|
+
}));
|
|
438
|
+
}
|
|
439
|
+
window.addEventListener("click", this.onClickOutside);
|
|
440
|
+
window.addEventListener("scroll", this.onWindowScroll, true);
|
|
441
|
+
}
|
|
442
|
+
/**
|
|
443
|
+
* @return {?}
|
|
444
|
+
*/
|
|
445
|
+
close() {
|
|
446
|
+
if (this.opened === "no") {
|
|
447
|
+
return;
|
|
448
|
+
}
|
|
449
|
+
this.opened = "no";
|
|
450
|
+
this.positioning = "no";
|
|
451
|
+
this.openChange.emit(false);
|
|
452
|
+
window.removeEventListener("click", this.onClickOutside);
|
|
453
|
+
window.removeEventListener("scroll", this.onWindowScroll, true);
|
|
454
|
+
this.ref.markForCheck();
|
|
455
|
+
}
|
|
456
|
+
/**
|
|
457
|
+
* @return {?}
|
|
458
|
+
*/
|
|
459
|
+
toggle() {
|
|
460
|
+
if (this.opened === "yes") {
|
|
461
|
+
this.close();
|
|
462
|
+
}
|
|
463
|
+
else {
|
|
464
|
+
this.open();
|
|
465
|
+
}
|
|
466
|
+
}
|
|
467
|
+
/**
|
|
468
|
+
* @param {?} event
|
|
469
|
+
* @return {?}
|
|
470
|
+
*/
|
|
471
|
+
onClickTrigger(event) {
|
|
472
|
+
event.stopPropagation();
|
|
473
|
+
this.toggle();
|
|
474
|
+
}
|
|
475
|
+
// 计算并设置位置(支持固定尺寸,可在打开前调用)
|
|
476
|
+
/**
|
|
477
|
+
* @private
|
|
478
|
+
* @return {?}
|
|
479
|
+
*/
|
|
480
|
+
calculateAndSetPosition() {
|
|
481
|
+
if (!this.currentTriggerElement) {
|
|
482
|
+
return;
|
|
483
|
+
}
|
|
484
|
+
/** @type {?} */
|
|
485
|
+
const triggerRect = this.currentTriggerElement.getBoundingClientRect();
|
|
486
|
+
/** @type {?} */
|
|
487
|
+
const viewportWidth = window.innerWidth;
|
|
488
|
+
/** @type {?} */
|
|
489
|
+
const viewportHeight = window.innerHeight;
|
|
490
|
+
// 获取弹窗尺寸(优先使用固定尺寸)
|
|
491
|
+
/** @type {?} */
|
|
492
|
+
const popupSize = {
|
|
493
|
+
width: this._width !== null
|
|
494
|
+
? this._width
|
|
495
|
+
: this.popupElement && this.popupElement.nativeElement
|
|
496
|
+
? this.popupElement.nativeElement.offsetWidth
|
|
497
|
+
: 300,
|
|
498
|
+
height: this._height !== null
|
|
499
|
+
? this._height
|
|
500
|
+
: this.popupElement && this.popupElement.nativeElement
|
|
501
|
+
? this.popupElement.nativeElement.offsetHeight
|
|
502
|
+
: 200,
|
|
503
|
+
};
|
|
504
|
+
// 创建虚拟的 popupRect(用于计算)
|
|
505
|
+
/** @type {?} */
|
|
506
|
+
const popupRect = (/** @type {?} */ ({
|
|
507
|
+
width: popupSize.width,
|
|
508
|
+
height: popupSize.height,
|
|
509
|
+
top: 0,
|
|
510
|
+
left: 0,
|
|
511
|
+
right: popupSize.width,
|
|
512
|
+
bottom: popupSize.height,
|
|
513
|
+
x: 0,
|
|
514
|
+
y: 0,
|
|
515
|
+
toJSON: (/**
|
|
516
|
+
* @return {?}
|
|
517
|
+
*/
|
|
518
|
+
() => ({})),
|
|
519
|
+
}));
|
|
520
|
+
// 获取目标位置信息
|
|
521
|
+
/** @type {?} */
|
|
522
|
+
const targetInfo = this.placementInfo;
|
|
523
|
+
/** @type {?} */
|
|
524
|
+
let finalPlacement = targetInfo.placement;
|
|
525
|
+
/** @type {?} */
|
|
526
|
+
let finalPosition = targetInfo.position;
|
|
527
|
+
if (this.autoAdjust) {
|
|
528
|
+
/** @type {?} */
|
|
529
|
+
const optimal = this.calculateOptimalPlacement(triggerRect, popupRect, viewportWidth, viewportHeight);
|
|
530
|
+
finalPlacement = optimal.placement;
|
|
531
|
+
finalPosition = optimal.position;
|
|
532
|
+
}
|
|
533
|
+
this.actualPlacement = finalPlacement;
|
|
534
|
+
this.actualPosition = finalPosition;
|
|
535
|
+
// 立即设置位置(即使元素还没完全渲染,也要先设置位置)
|
|
536
|
+
if (this.popupElement && this.popupElement.nativeElement) {
|
|
537
|
+
this.setPopupPosition(triggerRect, popupRect, finalPlacement, finalPosition, popupSize);
|
|
538
|
+
// 强制应用样式,确保位置立即生效
|
|
539
|
+
this.ref.detectChanges();
|
|
540
|
+
}
|
|
541
|
+
else {
|
|
542
|
+
// 如果元素还不存在,延迟设置位置
|
|
543
|
+
setTimeout((/**
|
|
544
|
+
* @return {?}
|
|
545
|
+
*/
|
|
546
|
+
() => {
|
|
547
|
+
if (this.popupElement && this.popupElement.nativeElement) {
|
|
548
|
+
this.setPopupPosition(triggerRect, popupRect, finalPlacement, finalPosition, popupSize);
|
|
549
|
+
this.ref.detectChanges();
|
|
550
|
+
}
|
|
551
|
+
}), 0);
|
|
552
|
+
}
|
|
553
|
+
this.ref.markForCheck();
|
|
554
|
+
}
|
|
555
|
+
/**
|
|
556
|
+
* @return {?}
|
|
557
|
+
*/
|
|
558
|
+
updatePopupPosition() {
|
|
559
|
+
if (!this.currentTriggerElement ||
|
|
560
|
+
!this.popupElement ||
|
|
561
|
+
!this.popupElement.nativeElement) {
|
|
562
|
+
return;
|
|
563
|
+
}
|
|
564
|
+
// 如果提供了固定尺寸,使用固定尺寸计算
|
|
565
|
+
if (this._width !== null && this._height !== null) {
|
|
566
|
+
this.calculateAndSetPosition();
|
|
567
|
+
return;
|
|
568
|
+
}
|
|
569
|
+
/** @type {?} */
|
|
570
|
+
const triggerRect = this.currentTriggerElement.getBoundingClientRect();
|
|
571
|
+
/** @type {?} */
|
|
572
|
+
const popupRect = this.popupElement.nativeElement.getBoundingClientRect();
|
|
573
|
+
/** @type {?} */
|
|
574
|
+
const viewportWidth = window.innerWidth;
|
|
575
|
+
/** @type {?} */
|
|
576
|
+
const viewportHeight = window.innerHeight;
|
|
577
|
+
// 获取目标位置信息
|
|
578
|
+
/** @type {?} */
|
|
579
|
+
const targetInfo = this.placementInfo;
|
|
580
|
+
/** @type {?} */
|
|
581
|
+
let finalPlacement = targetInfo.placement;
|
|
582
|
+
/** @type {?} */
|
|
583
|
+
let finalPosition = targetInfo.position;
|
|
584
|
+
if (this.autoAdjust) {
|
|
585
|
+
/** @type {?} */
|
|
586
|
+
const optimal = this.calculateOptimalPlacement(triggerRect, popupRect, viewportWidth, viewportHeight);
|
|
587
|
+
finalPlacement = optimal.placement;
|
|
588
|
+
finalPosition = optimal.position;
|
|
589
|
+
}
|
|
590
|
+
this.actualPlacement = finalPlacement;
|
|
591
|
+
this.actualPosition = finalPosition;
|
|
592
|
+
this.setPopupPosition(triggerRect, popupRect, finalPlacement, finalPosition);
|
|
593
|
+
this.ref.markForCheck();
|
|
594
|
+
}
|
|
595
|
+
/**
|
|
596
|
+
* @param {?} triggerRect
|
|
597
|
+
* @param {?} popupRect
|
|
598
|
+
* @param {?} viewportWidth
|
|
599
|
+
* @param {?} viewportHeight
|
|
600
|
+
* @return {?}
|
|
601
|
+
*/
|
|
602
|
+
calculateOptimalPlacement(triggerRect, popupRect, viewportWidth, viewportHeight) {
|
|
603
|
+
/** @type {?} */
|
|
604
|
+
const targetInfo = this.placementInfo;
|
|
605
|
+
/** @type {?} */
|
|
606
|
+
const preferred = targetInfo.placement;
|
|
607
|
+
/** @type {?} */
|
|
608
|
+
const preferredPosition = targetInfo.position;
|
|
609
|
+
/** @type {?} */
|
|
610
|
+
const space = {
|
|
611
|
+
top: triggerRect.top,
|
|
612
|
+
bottom: viewportHeight - triggerRect.bottom,
|
|
613
|
+
left: triggerRect.left,
|
|
614
|
+
right: viewportWidth - triggerRect.right,
|
|
615
|
+
};
|
|
616
|
+
/** @type {?} */
|
|
617
|
+
const popupSize = {
|
|
618
|
+
width: popupRect.width || 300,
|
|
619
|
+
// 默认宽度,如果还没有渲染
|
|
620
|
+
height: popupRect.height || 200,
|
|
621
|
+
};
|
|
622
|
+
// 检查首选位置是否有足够空间
|
|
623
|
+
/** @type {?} */
|
|
624
|
+
const hasEnoughSpace = (/**
|
|
625
|
+
* @param {?} placement
|
|
626
|
+
* @return {?}
|
|
627
|
+
*/
|
|
628
|
+
(placement) => {
|
|
629
|
+
switch (placement) {
|
|
630
|
+
case "top":
|
|
631
|
+
return space.top >= popupSize.height + this.offset;
|
|
632
|
+
case "bottom":
|
|
633
|
+
return space.bottom >= popupSize.height + this.offset;
|
|
634
|
+
case "left":
|
|
635
|
+
return space.left >= popupSize.width + this.offset;
|
|
636
|
+
case "right":
|
|
637
|
+
return space.right >= popupSize.width + this.offset;
|
|
638
|
+
}
|
|
639
|
+
});
|
|
640
|
+
// 检查指定位置是否有足够空间
|
|
641
|
+
/** @type {?} */
|
|
642
|
+
const checkPositionSpace = (/**
|
|
643
|
+
* @param {?} placement
|
|
644
|
+
* @param {?} position
|
|
645
|
+
* @return {?}
|
|
646
|
+
*/
|
|
647
|
+
(placement, position) => {
|
|
648
|
+
if (placement === "top" || placement === "bottom") {
|
|
649
|
+
// 对于上下位置,检查水平对齐空间
|
|
650
|
+
/** @type {?} */
|
|
651
|
+
const popupWidth = popupSize.width;
|
|
652
|
+
/** @type {?} */
|
|
653
|
+
const spaceLeft = triggerRect.left;
|
|
654
|
+
/** @type {?} */
|
|
655
|
+
const spaceRight = viewportWidth - triggerRect.right;
|
|
656
|
+
if (position === "center") {
|
|
657
|
+
return spaceLeft >= popupWidth / 2 && spaceRight >= popupWidth / 2;
|
|
658
|
+
}
|
|
659
|
+
else if (position === "start") {
|
|
660
|
+
return spaceLeft >= popupWidth;
|
|
661
|
+
}
|
|
662
|
+
else {
|
|
663
|
+
// end
|
|
664
|
+
return spaceRight >= popupWidth;
|
|
665
|
+
}
|
|
666
|
+
}
|
|
667
|
+
else {
|
|
668
|
+
// 对于左右位置,检查垂直对齐空间
|
|
669
|
+
/** @type {?} */
|
|
670
|
+
const popupHeight = popupSize.height;
|
|
671
|
+
/** @type {?} */
|
|
672
|
+
const spaceTop = triggerRect.top;
|
|
673
|
+
/** @type {?} */
|
|
674
|
+
const spaceBottom = viewportHeight - triggerRect.bottom;
|
|
675
|
+
if (position === "center") {
|
|
676
|
+
return spaceTop >= popupHeight / 2 && spaceBottom >= popupHeight / 2;
|
|
677
|
+
}
|
|
678
|
+
else if (position === "start") {
|
|
679
|
+
return spaceTop >= popupHeight;
|
|
680
|
+
}
|
|
681
|
+
else {
|
|
682
|
+
// end
|
|
683
|
+
return spaceBottom >= popupHeight;
|
|
684
|
+
}
|
|
685
|
+
}
|
|
686
|
+
});
|
|
687
|
+
// 计算位置是否合适(考虑水平/垂直对齐)
|
|
688
|
+
/** @type {?} */
|
|
689
|
+
const calculatePosition = (/**
|
|
690
|
+
* @param {?} placement
|
|
691
|
+
* @return {?}
|
|
692
|
+
*/
|
|
693
|
+
(placement) => {
|
|
694
|
+
if (placement === "top" || placement === "bottom") {
|
|
695
|
+
// 对于上下位置,检查水平对齐
|
|
696
|
+
/** @type {?} */
|
|
697
|
+
const popupWidth = popupSize.width;
|
|
698
|
+
/** @type {?} */
|
|
699
|
+
const spaceLeft = triggerRect.left;
|
|
700
|
+
/** @type {?} */
|
|
701
|
+
const spaceRight = viewportWidth - triggerRect.right;
|
|
702
|
+
// 优先尝试使用用户指定的position
|
|
703
|
+
if (checkPositionSpace(placement, preferredPosition)) {
|
|
704
|
+
return preferredPosition;
|
|
705
|
+
}
|
|
706
|
+
// 如果中心对齐有足够空间,优先使用中心
|
|
707
|
+
if (spaceLeft >= popupWidth / 2 && spaceRight >= popupWidth / 2) {
|
|
708
|
+
return "center";
|
|
709
|
+
}
|
|
710
|
+
// 如果左侧空间不足,使用右对齐
|
|
711
|
+
if (spaceLeft < popupWidth / 2) {
|
|
712
|
+
return "end";
|
|
713
|
+
}
|
|
714
|
+
// 如果右侧空间不足,使用左对齐
|
|
715
|
+
if (spaceRight < popupWidth / 2) {
|
|
716
|
+
return "start";
|
|
717
|
+
}
|
|
718
|
+
return preferredPosition;
|
|
719
|
+
}
|
|
720
|
+
else {
|
|
721
|
+
// 对于左右位置,检查垂直对齐
|
|
722
|
+
/** @type {?} */
|
|
723
|
+
const popupHeight = popupSize.height;
|
|
724
|
+
/** @type {?} */
|
|
725
|
+
const spaceTop = triggerRect.top;
|
|
726
|
+
/** @type {?} */
|
|
727
|
+
const spaceBottom = viewportHeight - triggerRect.bottom;
|
|
728
|
+
// 优先尝试使用用户指定的position
|
|
729
|
+
if (checkPositionSpace(placement, preferredPosition)) {
|
|
730
|
+
return preferredPosition;
|
|
731
|
+
}
|
|
732
|
+
// 如果中心对齐有足够空间,优先使用中心
|
|
733
|
+
if (spaceTop >= popupHeight / 2 && spaceBottom >= popupHeight / 2) {
|
|
734
|
+
return "center";
|
|
735
|
+
}
|
|
736
|
+
// 如果上方空间不足,使用下对齐
|
|
737
|
+
if (spaceTop < popupHeight / 2) {
|
|
738
|
+
return "end";
|
|
739
|
+
}
|
|
740
|
+
// 如果下方空间不足,使用上对齐
|
|
741
|
+
if (spaceBottom < popupHeight / 2) {
|
|
742
|
+
return "start";
|
|
743
|
+
}
|
|
744
|
+
return preferredPosition;
|
|
745
|
+
}
|
|
746
|
+
});
|
|
747
|
+
// 如果首选位置有足够空间,优先使用用户指定的position
|
|
748
|
+
if (hasEnoughSpace(preferred)) {
|
|
749
|
+
// 检查用户指定的position是否有足够空间
|
|
750
|
+
if (checkPositionSpace(preferred, preferredPosition)) {
|
|
751
|
+
return {
|
|
752
|
+
placement: preferred,
|
|
753
|
+
position: preferredPosition,
|
|
754
|
+
};
|
|
755
|
+
}
|
|
756
|
+
// 如果指定的position空间不足,自动计算最佳position
|
|
757
|
+
return {
|
|
758
|
+
placement: preferred,
|
|
759
|
+
position: calculatePosition(preferred),
|
|
760
|
+
};
|
|
761
|
+
}
|
|
762
|
+
// 根据首选位置选择备选位置
|
|
763
|
+
/** @type {?} */
|
|
764
|
+
const alternatives = {
|
|
765
|
+
bottom: ["top", "right", "left"],
|
|
766
|
+
top: ["bottom", "right", "left"],
|
|
767
|
+
right: ["left", "bottom", "top"],
|
|
768
|
+
left: ["right", "bottom", "top"],
|
|
769
|
+
};
|
|
770
|
+
/** @type {?} */
|
|
771
|
+
const altList = alternatives[preferred] || [];
|
|
772
|
+
// 尝试找到第一个有足够空间的备选位置
|
|
773
|
+
for (const alt of altList) {
|
|
774
|
+
if (hasEnoughSpace(alt)) {
|
|
775
|
+
return {
|
|
776
|
+
placement: alt,
|
|
777
|
+
position: calculatePosition(alt),
|
|
778
|
+
};
|
|
779
|
+
}
|
|
780
|
+
}
|
|
781
|
+
// 如果都没有足够空间,选择空间最大的位置
|
|
782
|
+
/** @type {?} */
|
|
783
|
+
const maxSpace = Math.max(space.top, space.bottom, space.left, space.right);
|
|
784
|
+
/** @type {?} */
|
|
785
|
+
let bestPlacement;
|
|
786
|
+
if (maxSpace === space.top)
|
|
787
|
+
bestPlacement = "top";
|
|
788
|
+
else if (maxSpace === space.bottom)
|
|
789
|
+
bestPlacement = "bottom";
|
|
790
|
+
else if (maxSpace === space.left)
|
|
791
|
+
bestPlacement = "left";
|
|
792
|
+
else
|
|
793
|
+
bestPlacement = "right";
|
|
794
|
+
return {
|
|
795
|
+
placement: bestPlacement,
|
|
796
|
+
position: calculatePosition(bestPlacement),
|
|
797
|
+
};
|
|
798
|
+
}
|
|
799
|
+
// 计算位置坐标(纯计算,不依赖DOM,包含边界约束)
|
|
800
|
+
/**
|
|
801
|
+
* @private
|
|
802
|
+
* @param {?} triggerRect
|
|
803
|
+
* @param {?} placement
|
|
804
|
+
* @param {?} position
|
|
805
|
+
* @param {?} popupSize
|
|
806
|
+
* @return {?}
|
|
807
|
+
*/
|
|
808
|
+
calculatePositionCoordinates(triggerRect, placement, position, popupSize) {
|
|
809
|
+
/** @type {?} */
|
|
810
|
+
const triggerWidth = triggerRect.width;
|
|
811
|
+
/** @type {?} */
|
|
812
|
+
const triggerHeight = triggerRect.height;
|
|
813
|
+
/** @type {?} */
|
|
814
|
+
const viewportWidth = window.innerWidth;
|
|
815
|
+
/** @type {?} */
|
|
816
|
+
const viewportHeight = window.innerHeight;
|
|
817
|
+
/** @type {?} */
|
|
818
|
+
const scrollX = window.scrollX;
|
|
819
|
+
/** @type {?} */
|
|
820
|
+
const scrollY = window.scrollY;
|
|
821
|
+
/** @type {?} */
|
|
822
|
+
let top = 0;
|
|
823
|
+
/** @type {?} */
|
|
824
|
+
let left = 0;
|
|
825
|
+
/** @type {?} */
|
|
826
|
+
let translateX = "0";
|
|
827
|
+
/** @type {?} */
|
|
828
|
+
let translateY = "0";
|
|
829
|
+
switch (placement) {
|
|
830
|
+
case "top":
|
|
831
|
+
top = triggerRect.top + scrollY - this.offset;
|
|
832
|
+
translateY = "-100%";
|
|
833
|
+
if (position === "start") {
|
|
834
|
+
left = triggerRect.left + scrollX;
|
|
835
|
+
}
|
|
836
|
+
else if (position === "center") {
|
|
837
|
+
left = triggerRect.left + scrollX + triggerWidth / 2;
|
|
838
|
+
translateX = "-50%";
|
|
839
|
+
}
|
|
840
|
+
else {
|
|
841
|
+
left = triggerRect.left + scrollX + triggerWidth;
|
|
842
|
+
translateX = "-100%";
|
|
843
|
+
}
|
|
844
|
+
// 边界约束:确保弹窗不超出视口
|
|
845
|
+
if (top - popupSize.height < scrollY) {
|
|
846
|
+
top = scrollY + 8; // 距离顶部至少8px
|
|
847
|
+
}
|
|
848
|
+
if (left < scrollX) {
|
|
849
|
+
left = scrollX + 8;
|
|
850
|
+
translateX = "0";
|
|
851
|
+
}
|
|
852
|
+
else if (left + popupSize.width > scrollX + viewportWidth) {
|
|
853
|
+
left = scrollX + viewportWidth - popupSize.width - 8;
|
|
854
|
+
translateX = "0";
|
|
855
|
+
}
|
|
856
|
+
break;
|
|
857
|
+
case "bottom":
|
|
858
|
+
top = triggerRect.bottom + scrollY + this.offset;
|
|
859
|
+
if (position === "start") {
|
|
860
|
+
left = triggerRect.left + scrollX;
|
|
861
|
+
}
|
|
862
|
+
else if (position === "center") {
|
|
863
|
+
left = triggerRect.left + scrollX + triggerWidth / 2;
|
|
864
|
+
translateX = "-50%";
|
|
865
|
+
}
|
|
866
|
+
else {
|
|
867
|
+
left = triggerRect.left + scrollX + triggerWidth;
|
|
868
|
+
translateX = "-100%";
|
|
869
|
+
}
|
|
870
|
+
// 边界约束:确保弹窗不超出视口
|
|
871
|
+
if (top + popupSize.height > scrollY + viewportHeight) {
|
|
872
|
+
top = scrollY + viewportHeight - popupSize.height - 8; // 距离底部至少8px
|
|
873
|
+
}
|
|
874
|
+
if (left < scrollX) {
|
|
875
|
+
left = scrollX + 8;
|
|
876
|
+
translateX = "0";
|
|
877
|
+
}
|
|
878
|
+
else if (left + popupSize.width > scrollX + viewportWidth) {
|
|
879
|
+
left = scrollX + viewportWidth - popupSize.width - 8;
|
|
880
|
+
translateX = "0";
|
|
881
|
+
}
|
|
882
|
+
break;
|
|
883
|
+
case "left":
|
|
884
|
+
left = triggerRect.left + scrollX - this.offset;
|
|
885
|
+
translateX = "-100%";
|
|
886
|
+
if (position === "start") {
|
|
887
|
+
top = triggerRect.top + scrollY;
|
|
888
|
+
}
|
|
889
|
+
else if (position === "center") {
|
|
890
|
+
top = triggerRect.top + scrollY + triggerHeight / 2;
|
|
891
|
+
translateY = "-50%";
|
|
892
|
+
}
|
|
893
|
+
else {
|
|
894
|
+
top = triggerRect.top + scrollY + triggerHeight;
|
|
895
|
+
translateY = "-100%";
|
|
896
|
+
}
|
|
897
|
+
// 边界约束:确保弹窗不超出视口
|
|
898
|
+
if (left - popupSize.width < scrollX) {
|
|
899
|
+
left = scrollX + 8; // 距离左侧至少8px
|
|
900
|
+
}
|
|
901
|
+
if (top < scrollY) {
|
|
902
|
+
top = scrollY + 8;
|
|
903
|
+
translateY = "0";
|
|
904
|
+
}
|
|
905
|
+
else if (top + popupSize.height > scrollY + viewportHeight) {
|
|
906
|
+
top = scrollY + viewportHeight - popupSize.height - 8;
|
|
907
|
+
translateY = "0";
|
|
908
|
+
}
|
|
909
|
+
break;
|
|
910
|
+
case "right":
|
|
911
|
+
left = triggerRect.right + scrollX + this.offset;
|
|
912
|
+
if (position === "start") {
|
|
913
|
+
top = triggerRect.top + scrollY;
|
|
914
|
+
}
|
|
915
|
+
else if (position === "center") {
|
|
916
|
+
top = triggerRect.top + scrollY + triggerHeight / 2;
|
|
917
|
+
translateY = "-50%";
|
|
918
|
+
}
|
|
919
|
+
else {
|
|
920
|
+
top = triggerRect.top + scrollY + triggerHeight;
|
|
921
|
+
translateY = "-100%";
|
|
922
|
+
}
|
|
923
|
+
// 边界约束:确保弹窗不超出视口
|
|
924
|
+
if (left + popupSize.width > scrollX + viewportWidth) {
|
|
925
|
+
left = scrollX + viewportWidth - popupSize.width - 8; // 距离右侧至少8px
|
|
926
|
+
}
|
|
927
|
+
if (top < scrollY) {
|
|
928
|
+
top = scrollY + 8;
|
|
929
|
+
translateY = "0";
|
|
930
|
+
}
|
|
931
|
+
else if (top + popupSize.height > scrollY + viewportHeight) {
|
|
932
|
+
top = scrollY + viewportHeight - popupSize.height - 8;
|
|
933
|
+
translateY = "0";
|
|
934
|
+
}
|
|
935
|
+
break;
|
|
936
|
+
}
|
|
937
|
+
return {
|
|
938
|
+
top: top + "px",
|
|
939
|
+
left: left + "px",
|
|
940
|
+
translateX,
|
|
941
|
+
translateY,
|
|
942
|
+
};
|
|
943
|
+
}
|
|
944
|
+
/**
|
|
945
|
+
* @param {?} triggerRect
|
|
946
|
+
* @param {?} popupRect
|
|
947
|
+
* @param {?} placement
|
|
948
|
+
* @param {?} position
|
|
949
|
+
* @param {?=} popupSize
|
|
950
|
+
* @return {?}
|
|
951
|
+
*/
|
|
952
|
+
setPopupPosition(triggerRect, popupRect, placement, position, popupSize) {
|
|
953
|
+
/** @type {?} */
|
|
954
|
+
const popupEl = this.popupElement && this.popupElement.nativeElement
|
|
955
|
+
? this.popupElement.nativeElement
|
|
956
|
+
: null;
|
|
957
|
+
if (!popupEl) {
|
|
958
|
+
return; // 如果元素还不存在,无法设置位置
|
|
959
|
+
}
|
|
960
|
+
// 优先使用传入的固定尺寸,否则从DOM获取
|
|
961
|
+
/** @type {?} */
|
|
962
|
+
const popupWidth = popupSize && popupSize.width !== null && popupSize.width !== undefined
|
|
963
|
+
? popupSize.width
|
|
964
|
+
: popupRect.width || popupEl.offsetWidth || 300;
|
|
965
|
+
/** @type {?} */
|
|
966
|
+
const popupHeight = popupSize && popupSize.height !== null && popupSize.height !== undefined
|
|
967
|
+
? popupSize.height
|
|
968
|
+
: popupRect.height || popupEl.offsetHeight || 200;
|
|
969
|
+
/** @type {?} */
|
|
970
|
+
const coordinates = this.calculatePositionCoordinates(triggerRect, placement, position, { width: popupWidth, height: popupHeight });
|
|
971
|
+
popupEl.style.top = coordinates.top;
|
|
972
|
+
popupEl.style.left = coordinates.left;
|
|
973
|
+
popupEl.style.transform = `translate(${coordinates.translateX}, ${coordinates.translateY})`;
|
|
974
|
+
}
|
|
975
|
+
/**
|
|
976
|
+
* @return {?}
|
|
977
|
+
*/
|
|
978
|
+
onWindowResize() {
|
|
979
|
+
if (this.opened === "yes") {
|
|
980
|
+
this.updatePopupPosition();
|
|
981
|
+
}
|
|
982
|
+
}
|
|
983
|
+
}
|
|
984
|
+
SmartPopupComponent.decorators = [
|
|
985
|
+
{ type: Component, args: [{
|
|
986
|
+
selector: "rs-smart-popup",
|
|
987
|
+
template: "<!-- \u89E6\u53D1\u5143\u7D20\u63D2\u69FD -->\r\n<ng-container #trigger>\r\n <ng-content select=\"[trigger]\"></ng-content>\r\n</ng-container>\r\n\r\n<!-- \u5F39\u7A97\u5185\u5BB9 -->\r\n<div\r\n #popup\r\n class=\"rs-smart-popup-content\"\r\n [attr.data-opened]=\"opened\"\r\n [attr.data-positioning]=\"positioning\"\r\n>\r\n <div *ngIf=\"loading\" class=\"rs-smart-popup-loading\">\r\n <div class=\"loading-spinner\"></div>\r\n </div>\r\n <div *ngIf=\"!loading\" class=\"rs-smart-popup-body\">\r\n <ng-content></ng-content>\r\n </div>\r\n</div>\r\n\r\n",
|
|
988
|
+
styles: [".rs-smart-popup-trigger{display:inline-block;cursor:pointer}::ng-deep #rs-smart-popup-fixed-container{width:0;height:0;z-index:100000;pointer-events:none}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content{pointer-events:auto;position:fixed;z-index:100001;padding:8px;border-radius:8px;background:#fff;box-shadow:0 0 8px 0 rgba(0,0,0,.25);min-width:200px;max-width:400px;max-height:500px;overflow:auto}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content[data-opened=no]{display:none}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content[data-opened=yes]{display:block}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content[data-opened=yes][data-positioning=yes]{display:none!important}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content[data-opened=yes][data-positioning=no]{display:block;visibility:visible;opacity:1;pointer-events:auto;-webkit-animation:.2s ease-out popup-fade-in;animation:.2s ease-out popup-fade-in}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content .rs-smart-popup-loading{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:20px;min-height:100px}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content .rs-smart-popup-loading .loading-spinner{width:24px;height:24px;border:3px solid #f3f3f3;border-top:3px solid #3498db;border-radius:50%;-webkit-animation:1s linear infinite spin;animation:1s linear infinite spin;margin-bottom:12px}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content .rs-smart-popup-loading .loading-text{color:#666;font-size:14px}@-webkit-keyframes spin{0%{transform:rotate(0)}100%{transform:rotate(360deg)}}@keyframes spin{0%{transform:rotate(0)}100%{transform:rotate(360deg)}}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content .rs-smart-popup-body{width:100%}@-webkit-keyframes popup-fade-in{from{opacity:0;transform:translate(var(--translate-x,0),var(--translate-y,0)) scale(.95)}to{opacity:1;transform:translate(var(--translate-x,0),var(--translate-y,0)) scale(1)}}@keyframes popup-fade-in{from{opacity:0;transform:translate(var(--translate-x,0),var(--translate-y,0)) scale(.95)}to{opacity:1;transform:translate(var(--translate-x,0),var(--translate-y,0)) scale(1)}}"]
|
|
989
|
+
}] }
|
|
990
|
+
];
|
|
991
|
+
/** @nocollapse */
|
|
992
|
+
SmartPopupComponent.ctorParameters = () => [
|
|
993
|
+
{ type: ChangeDetectorRef }
|
|
994
|
+
];
|
|
995
|
+
SmartPopupComponent.propDecorators = {
|
|
996
|
+
_placement: [{ type: Input, args: ["placement",] }],
|
|
997
|
+
_offset: [{ type: Input, args: ["offset",] }],
|
|
998
|
+
_autoAdjust: [{ type: Input, args: ["autoAdjust",] }],
|
|
999
|
+
triggerElementRef: [{ type: Input, args: ["triggerElement",] }],
|
|
1000
|
+
loading: [{ type: Input, args: ["loading",] }],
|
|
1001
|
+
_open: [{ type: Input, args: ["open",] }],
|
|
1002
|
+
_width: [{ type: Input, args: ["width",] }],
|
|
1003
|
+
_height: [{ type: Input, args: ["height",] }],
|
|
1004
|
+
openChange: [{ type: Output, args: ["openChange",] }],
|
|
1005
|
+
contentLoad: [{ type: Output, args: ["contentLoad",] }],
|
|
1006
|
+
internalTriggerElement: [{ type: ViewChild, args: ["trigger", { static: false },] }],
|
|
1007
|
+
popupElement: [{ type: ViewChild, args: ["popup", { static: false },] }],
|
|
1008
|
+
onWindowResize: [{ type: HostListener, args: ["window:resize",] }]
|
|
1009
|
+
};
|
|
1010
|
+
if (false) {
|
|
1011
|
+
/** @type {?} */
|
|
1012
|
+
SmartPopupComponent.prototype._placement;
|
|
1013
|
+
/** @type {?} */
|
|
1014
|
+
SmartPopupComponent.prototype._offset;
|
|
1015
|
+
/** @type {?} */
|
|
1016
|
+
SmartPopupComponent.prototype._autoAdjust;
|
|
1017
|
+
/** @type {?} */
|
|
1018
|
+
SmartPopupComponent.prototype.triggerElementRef;
|
|
1019
|
+
/** @type {?} */
|
|
1020
|
+
SmartPopupComponent.prototype.loading;
|
|
1021
|
+
/** @type {?} */
|
|
1022
|
+
SmartPopupComponent.prototype._open;
|
|
1023
|
+
/** @type {?} */
|
|
1024
|
+
SmartPopupComponent.prototype._width;
|
|
1025
|
+
/** @type {?} */
|
|
1026
|
+
SmartPopupComponent.prototype._height;
|
|
1027
|
+
/** @type {?} */
|
|
1028
|
+
SmartPopupComponent.prototype.openChange;
|
|
1029
|
+
/** @type {?} */
|
|
1030
|
+
SmartPopupComponent.prototype.contentLoad;
|
|
1031
|
+
/** @type {?} */
|
|
1032
|
+
SmartPopupComponent.prototype.internalTriggerElement;
|
|
1033
|
+
/** @type {?} */
|
|
1034
|
+
SmartPopupComponent.prototype.popupElement;
|
|
1035
|
+
/** @type {?} */
|
|
1036
|
+
SmartPopupComponent.prototype.currentTriggerElement;
|
|
1037
|
+
/** @type {?} */
|
|
1038
|
+
SmartPopupComponent.prototype.FIXED_CONTAINER_ID;
|
|
1039
|
+
/** @type {?} */
|
|
1040
|
+
SmartPopupComponent.prototype.fixedContainerEl;
|
|
1041
|
+
/** @type {?} */
|
|
1042
|
+
SmartPopupComponent.prototype.opened;
|
|
1043
|
+
/** @type {?} */
|
|
1044
|
+
SmartPopupComponent.prototype.positioning;
|
|
1045
|
+
/** @type {?} */
|
|
1046
|
+
SmartPopupComponent.prototype.actualPlacement;
|
|
1047
|
+
/** @type {?} */
|
|
1048
|
+
SmartPopupComponent.prototype.actualPosition;
|
|
1049
|
+
/** @type {?} */
|
|
1050
|
+
SmartPopupComponent.prototype.onClickOutside;
|
|
1051
|
+
/** @type {?} */
|
|
1052
|
+
SmartPopupComponent.prototype.onWindowScroll;
|
|
1053
|
+
/**
|
|
1054
|
+
* @type {?}
|
|
1055
|
+
* @private
|
|
1056
|
+
*/
|
|
1057
|
+
SmartPopupComponent.prototype.ref;
|
|
1058
|
+
}
|
|
1059
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.component.js","sourceRoot":"ng://raise-common-lib/","sources":["lib/smart-popup/index.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  AfterViewInit,\r\n  ChangeDetectorRef,\r\n  Component,\r\n  ElementRef,\r\n  EventEmitter,\r\n  Input,\r\n  OnDestroy,\r\n  OnChanges,\r\n  SimpleChanges,\r\n  Output,\r\n  ViewChild,\r\n  HostListener,\r\n} from \"@angular/core\";\r\n\r\nexport type PlacementType =\r\n  | \"TopLeft\"\r\n  | \"TopCenter\"\r\n  | \"TopRight\"\r\n  | \"BottomLeft\"\r\n  | \"BottomCenter\"\r\n  | \"BottomRight\"\r\n  | \"LeftTop\"\r\n  | \"LeftCenter\"\r\n  | \"LeftBottom\"\r\n  | \"RightTop\"\r\n  | \"RightCenter\"\r\n  | \"RightBottom\";\r\n\r\n@Component({\r\n  selector: \"rs-smart-popup\",\r\n  templateUrl: \"./index.component.html\",\r\n  styleUrls: [\"./index.component.scss\"],\r\n})\r\nexport class SmartPopupComponent\r\n  implements AfterViewInit, OnDestroy, OnChanges\r\n{\r\n  @Input(\"placement\") _placement: PlacementType = \"BottomLeft\";\r\n  @Input(\"offset\") _offset: number = 8;\r\n  @Input(\"autoAdjust\") _autoAdjust: \"yes\" | \"no\" = \"yes\";\r\n  @Input(\"triggerElement\") triggerElementRef: HTMLElement | null = null;\r\n  @Input(\"loading\") loading: boolean = false;\r\n  @Input(\"open\") _open: boolean = false;\r\n  @Input(\"width\") _width: number | null = null;\r\n  @Input(\"height\") _height: number | null = null;\r\n\r\n  // 解析placement格式\r\n  get placementInfo(): {\r\n    placement: \"top\" | \"left\" | \"right\" | \"bottom\";\r\n    position: \"start\" | \"center\" | \"end\";\r\n  } {\r\n    const placementMap: {\r\n      [key: string]: {\r\n        placement: \"top\" | \"left\" | \"right\" | \"bottom\";\r\n        position: \"start\" | \"center\" | \"end\";\r\n      };\r\n    } = {\r\n      TopLeft: { placement: \"top\", position: \"start\" },\r\n      TopCenter: { placement: \"top\", position: \"center\" },\r\n      TopRight: { placement: \"top\", position: \"end\" },\r\n      BottomLeft: { placement: \"bottom\", position: \"start\" },\r\n      BottomCenter: { placement: \"bottom\", position: \"center\" },\r\n      BottomRight: { placement: \"bottom\", position: \"end\" },\r\n      LeftTop: { placement: \"left\", position: \"start\" },\r\n      LeftCenter: { placement: \"left\", position: \"center\" },\r\n      LeftBottom: { placement: \"left\", position: \"end\" },\r\n      RightTop: { placement: \"right\", position: \"start\" },\r\n      RightCenter: { placement: \"right\", position: \"center\" },\r\n      RightBottom: { placement: \"right\", position: \"end\" },\r\n    };\r\n    return (\r\n      placementMap[this._placement] || {\r\n        placement: \"bottom\",\r\n        position: \"start\",\r\n      }\r\n    );\r\n  }\r\n\r\n  get offset() {\r\n    return this._offset || 8;\r\n  }\r\n\r\n  get autoAdjust() {\r\n    return this._autoAdjust === \"yes\";\r\n  }\r\n\r\n  @Output(\"openChange\") openChange = new EventEmitter<boolean>();\r\n  @Output(\"contentLoad\") contentLoad = new EventEmitter<void>();\r\n\r\n  @ViewChild(\"trigger\", { static: false })\r\n  internalTriggerElement: ElementRef<HTMLElement>;\r\n  @ViewChild(\"popup\", { static: false })\r\n  popupElement: ElementRef<HTMLDivElement>;\r\n\r\n  currentTriggerElement: HTMLElement | null = null;\r\n\r\n  constructor(private ref: ChangeDetectorRef) {}\r\n\r\n  ngAfterViewInit() {\r\n    this.fixedContainerEl = this.getFixedContainer();\r\n    this.fixedContainerEl.append(this.popupElement.nativeElement);\r\n    this.updateTriggerElement();\r\n\r\n    // 如果提供了固定尺寸，立即应用\r\n    if (\r\n      this._width !== null &&\r\n      this._height !== null &&\r\n      this.popupElement &&\r\n      this.popupElement.nativeElement\r\n    ) {\r\n      const popupEl = this.popupElement.nativeElement;\r\n      popupEl.style.width = this._width + \"px\";\r\n      popupEl.style.height = this._height + \"px\";\r\n      popupEl.style.minWidth = this._width + \"px\";\r\n      popupEl.style.maxWidth = this._width + \"px\";\r\n      popupEl.style.minHeight = this._height + \"px\";\r\n      popupEl.style.maxHeight = this._height + \"px\";\r\n    }\r\n\r\n    // 如果外部传入 open 属性为 true，自动打开\r\n    if (this._open) {\r\n      setTimeout(() => this.open(), 0);\r\n    }\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges) {\r\n    // 处理固定尺寸变化\r\n    if (\r\n      (changes._width || changes._height) &&\r\n      this.popupElement &&\r\n      this.popupElement.nativeElement\r\n    ) {\r\n      const popupEl = this.popupElement.nativeElement;\r\n      if (this._width !== null) {\r\n        popupEl.style.width = this._width + \"px\";\r\n        popupEl.style.minWidth = this._width + \"px\";\r\n        popupEl.style.maxWidth = this._width + \"px\";\r\n      }\r\n      if (this._height !== null) {\r\n        popupEl.style.height = this._height + \"px\";\r\n        popupEl.style.minHeight = this._height + \"px\";\r\n        popupEl.style.maxHeight = this._height + \"px\";\r\n      }\r\n    }\r\n\r\n    // 先处理触发元素变化（确保在打开前更新触发元素）\r\n    if (changes.triggerElementRef) {\r\n      const previousTrigger = this.currentTriggerElement;\r\n      this.updateTriggerElement();\r\n\r\n      // 如果弹窗已经打开，且触发元素发生了变化，立即更新位置\r\n      if (\r\n        this.opened === \"yes\" &&\r\n        previousTrigger !== this.currentTriggerElement &&\r\n        this.currentTriggerElement\r\n      ) {\r\n        // 使用新的打开方式更新位置\r\n        this.positioning = \"yes\";\r\n        this.ref.markForCheck();\r\n        requestAnimationFrame(() => {\r\n          requestAnimationFrame(() => {\r\n            this.updatePopupPosition();\r\n            this.positioning = \"no\";\r\n            this.ref.markForCheck();\r\n          });\r\n        });\r\n      }\r\n    }\r\n\r\n    // 处理 open 变化（确保在触发元素更新后处理）\r\n    if (changes._open) {\r\n      if (this._open && this.opened === \"no\") {\r\n        // 确保触发元素已更新\r\n        this.updateTriggerElement();\r\n        // 使用 setTimeout 确保所有变更都已处理\r\n        setTimeout(() => {\r\n          // 再次确保触发元素已更新\r\n          this.updateTriggerElement();\r\n          this.open();\r\n        }, 0);\r\n      } else if (!this._open && this.opened === \"yes\") {\r\n        this.close();\r\n      }\r\n    }\r\n\r\n    // 当加载完成时，更新位置（如果提供了固定尺寸，不需要更新位置）\r\n    if (\r\n      changes.loading &&\r\n      !changes.loading.currentValue &&\r\n      changes.loading.previousValue\r\n    ) {\r\n      if (this.opened === \"yes\") {\r\n        // 如果提供了固定尺寸，位置已经计算好了，不需要更新\r\n        if (this._width !== null && this._height !== null) {\r\n          this.contentLoad.emit();\r\n        } else {\r\n          // 如果正在定位，不需要更新位置\r\n          if (this.positioning === \"no\") {\r\n            requestAnimationFrame(() => {\r\n              this.updatePopupPosition();\r\n              this.contentLoad.emit();\r\n            });\r\n          } else {\r\n            // 如果正在定位，等待定位完成后再触发事件\r\n            const checkPositioning = () => {\r\n              if (this.positioning === \"no\") {\r\n                this.contentLoad.emit();\r\n              } else {\r\n                setTimeout(checkPositioning, 10);\r\n              }\r\n            };\r\n            checkPositioning();\r\n          }\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  ngOnDestroy() {\r\n    if (this.popupElement && this.popupElement.nativeElement) {\r\n      this.popupElement.nativeElement.remove();\r\n    }\r\n    window.removeEventListener(\"click\", this.onClickOutside);\r\n    window.removeEventListener(\"scroll\", this.onWindowScroll, true);\r\n  }\r\n\r\n  updateTriggerElement() {\r\n    if (this.triggerElementRef) {\r\n      this.currentTriggerElement = this.triggerElementRef;\r\n    } else if (\r\n      this.internalTriggerElement &&\r\n      this.internalTriggerElement.nativeElement\r\n    ) {\r\n      this.currentTriggerElement = this.internalTriggerElement.nativeElement;\r\n    } else {\r\n      this.currentTriggerElement = null;\r\n    }\r\n  }\r\n\r\n  FIXED_CONTAINER_ID = \"rs-smart-popup-fixed-container\";\r\n  fixedContainerEl: HTMLElement;\r\n  getFixedContainer() {\r\n    let containerElement = document.getElementById(this.FIXED_CONTAINER_ID);\r\n    if (!containerElement) {\r\n      containerElement = document.createElement(\"div\");\r\n      containerElement.setAttribute(\"id\", this.FIXED_CONTAINER_ID);\r\n      document.body.append(containerElement);\r\n    }\r\n    return containerElement;\r\n  }\r\n\r\n  opened: \"yes\" | \"no\" = \"no\";\r\n  positioning: \"yes\" | \"no\" = \"no\";\r\n  actualPlacement: \"top\" | \"left\" | \"right\" | \"bottom\" = \"bottom\";\r\n  actualPosition: \"start\" | \"center\" | \"end\" = \"start\";\r\n\r\n  open(triggerElement?: HTMLElement) {\r\n    // 如果传入了触发元素，使用它\r\n    if (triggerElement) {\r\n      this.currentTriggerElement = triggerElement;\r\n    } else {\r\n      this.updateTriggerElement();\r\n    }\r\n\r\n    if (!this.currentTriggerElement) {\r\n      console.warn(\"SmartPopupComponent: No trigger element found\");\r\n      return;\r\n    }\r\n\r\n    // 如果已经打开，先关闭再打开（更新位置）\r\n    if (this.opened === \"yes\") {\r\n      // 先关闭弹窗\r\n      this.opened = \"no\";\r\n      this.positioning = \"no\";\r\n      this.ref.markForCheck();\r\n      // 立即在新位置打开\r\n      requestAnimationFrame(() => {\r\n        // 再次确保触发元素已更新\r\n        this.updateTriggerElement();\r\n        this.openPopup();\r\n      });\r\n      return;\r\n    }\r\n\r\n    // 确保触发元素已更新\r\n    this.updateTriggerElement();\r\n    this.openPopup();\r\n  }\r\n\r\n  private openPopup() {\r\n    // 再次确保触发元素已更新\r\n    this.updateTriggerElement();\r\n\r\n    if (!this.currentTriggerElement) {\r\n      console.warn(\r\n        \"SmartPopupComponent: No trigger element found when opening popup\"\r\n      );\r\n      return;\r\n    }\r\n\r\n    // 如果提供了固定尺寸，可以在打开前就计算好位置\r\n    if (this._width !== null && this._height !== null) {\r\n      // 先计算位置（不依赖DOM渲染，直接根据triggerElement计算）\r\n      const triggerRect = this.currentTriggerElement.getBoundingClientRect();\r\n      const viewportWidth = window.innerWidth;\r\n      const viewportHeight = window.innerHeight;\r\n\r\n      const popupSize = {\r\n        width: this._width,\r\n        height: this._height,\r\n      };\r\n\r\n      const popupRect: DOMRect = {\r\n        width: popupSize.width,\r\n        height: popupSize.height,\r\n        top: 0,\r\n        left: 0,\r\n        right: popupSize.width,\r\n        bottom: popupSize.height,\r\n        x: 0,\r\n        y: 0,\r\n        toJSON: () => ({}),\r\n      } as DOMRect;\r\n\r\n      const targetInfo = this.placementInfo;\r\n      let finalPlacement = targetInfo.placement;\r\n      let finalPosition = targetInfo.position;\r\n\r\n      if (this.autoAdjust) {\r\n        const optimal = this.calculateOptimalPlacement(\r\n          triggerRect,\r\n          popupRect,\r\n          viewportWidth,\r\n          viewportHeight\r\n        );\r\n        finalPlacement = optimal.placement;\r\n        finalPosition = optimal.position;\r\n      }\r\n\r\n      this.actualPlacement = finalPlacement;\r\n      this.actualPosition = finalPosition;\r\n\r\n      // 计算位置坐标（不依赖DOM，直接计算）\r\n      const position = this.calculatePositionCoordinates(\r\n        triggerRect,\r\n        finalPlacement,\r\n        finalPosition,\r\n        popupSize\r\n      );\r\n\r\n      // 先设置打开状态，但保持隐藏（positioning = yes）\r\n      this.opened = \"yes\";\r\n      this.positioning = \"yes\";\r\n      this.openChange.emit(true);\r\n      this.ref.markForCheck();\r\n\r\n      // 强制更新视图，确保DOM已更新\r\n      this.ref.detectChanges();\r\n\r\n      // 如果元素已经存在，立即设置位置（不等待 requestAnimationFrame）\r\n      if (this.popupElement && this.popupElement.nativeElement) {\r\n        const popupEl = this.popupElement.nativeElement;\r\n\r\n        // 立即设置尺寸和位置\r\n        popupEl.style.width = this._width + \"px\";\r\n        popupEl.style.height = this._height + \"px\";\r\n        popupEl.style.minWidth = this._width + \"px\";\r\n        popupEl.style.maxWidth = this._width + \"px\";\r\n        popupEl.style.minHeight = this._height + \"px\";\r\n        popupEl.style.maxHeight = this._height + \"px\";\r\n        popupEl.style.top = position.top;\r\n        popupEl.style.left = position.left;\r\n        popupEl.style.transform = `translate(${position.translateX}, ${position.translateY})`;\r\n\r\n        // 强制同步应用样式\r\n        popupEl.offsetHeight;\r\n\r\n        // 位置设置完成后，立即显示弹窗\r\n        this.positioning = \"no\";\r\n        this.ref.markForCheck();\r\n      } else {\r\n        // 如果元素还不存在，等待渲染\r\n        requestAnimationFrame(() => {\r\n          if (this.popupElement && this.popupElement.nativeElement) {\r\n            const popupEl = this.popupElement.nativeElement;\r\n\r\n            // 立即设置尺寸和位置\r\n            popupEl.style.width = this._width + \"px\";\r\n            popupEl.style.height = this._height + \"px\";\r\n            popupEl.style.minWidth = this._width + \"px\";\r\n            popupEl.style.maxWidth = this._width + \"px\";\r\n            popupEl.style.minHeight = this._height + \"px\";\r\n            popupEl.style.maxHeight = this._height + \"px\";\r\n            popupEl.style.top = position.top;\r\n            popupEl.style.left = position.left;\r\n            popupEl.style.transform = `translate(${position.translateX}, ${position.translateY})`;\r\n\r\n            // 强制同步应用样式\r\n            popupEl.offsetHeight;\r\n\r\n            // 位置设置完成后，立即显示弹窗\r\n            this.positioning = \"no\";\r\n            this.ref.markForCheck();\r\n          }\r\n        });\r\n      }\r\n    } else {\r\n      // 没有固定尺寸，需要等待DOM渲染\r\n      this.positioning = \"yes\";\r\n      this.ref.markForCheck();\r\n\r\n      // 立即强制更新视图，确保DOM已更新\r\n      this.ref.detectChanges();\r\n\r\n      // 使用 requestAnimationFrame 确保在浏览器渲染之前更新位置\r\n      requestAnimationFrame(() => {\r\n        // 确保元素已渲染并获取尺寸\r\n        if (this.popupElement && this.popupElement.nativeElement) {\r\n          // 强制重新计算布局，确保元素尺寸已计算\r\n          this.popupElement.nativeElement.offsetHeight;\r\n\r\n          // 计算并设置位置\r\n          this.updatePopupPosition();\r\n\r\n          // 位置设置完成后，移除定位状态，显示弹窗\r\n          this.positioning = \"no\";\r\n          this.ref.markForCheck();\r\n        }\r\n      });\r\n    }\r\n\r\n    window.addEventListener(\"click\", this.onClickOutside);\r\n    window.addEventListener(\"scroll\", this.onWindowScroll, true);\r\n  }\r\n\r\n  close() {\r\n    if (this.opened === \"no\") {\r\n      return;\r\n    }\r\n\r\n    this.opened = \"no\";\r\n    this.positioning = \"no\";\r\n    this.openChange.emit(false);\r\n    window.removeEventListener(\"click\", this.onClickOutside);\r\n    window.removeEventListener(\"scroll\", this.onWindowScroll, true);\r\n    this.ref.markForCheck();\r\n  }\r\n\r\n  toggle() {\r\n    if (this.opened === \"yes\") {\r\n      this.close();\r\n    } else {\r\n      this.open();\r\n    }\r\n  }\r\n\r\n  onClickTrigger(event: MouseEvent) {\r\n    event.stopPropagation();\r\n    this.toggle();\r\n  }\r\n\r\n  onClickOutside = (event: MouseEvent) => {\r\n    const target = event.target as Element;\r\n    const isClickOnTrigger =\r\n      this.currentTriggerElement && this.currentTriggerElement.contains(target);\r\n    const isClickOnPopup =\r\n      this.popupElement &&\r\n      this.popupElement.nativeElement &&\r\n      this.popupElement.nativeElement.contains(target);\r\n\r\n    if (!isClickOnTrigger && !isClickOnPopup) {\r\n      this.close();\r\n    }\r\n  };\r\n\r\n  onWindowScroll = () => {\r\n    if (this.opened === \"yes\") {\r\n      this.updatePopupPosition();\r\n    }\r\n  };\r\n\r\n  // 计算并设置位置（支持固定尺寸，可在打开前调用）\r\n  private calculateAndSetPosition() {\r\n    if (!this.currentTriggerElement) {\r\n      return;\r\n    }\r\n\r\n    const triggerRect = this.currentTriggerElement.getBoundingClientRect();\r\n    const viewportWidth = window.innerWidth;\r\n    const viewportHeight = window.innerHeight;\r\n\r\n    // 获取弹窗尺寸（优先使用固定尺寸）\r\n    const popupSize = {\r\n      width:\r\n        this._width !== null\r\n          ? this._width\r\n          : this.popupElement && this.popupElement.nativeElement\r\n          ? this.popupElement.nativeElement.offsetWidth\r\n          : 300,\r\n      height:\r\n        this._height !== null\r\n          ? this._height\r\n          : this.popupElement && this.popupElement.nativeElement\r\n          ? this.popupElement.nativeElement.offsetHeight\r\n          : 200,\r\n    };\r\n\r\n    // 创建虚拟的 popupRect（用于计算）\r\n    const popupRect: DOMRect = {\r\n      width: popupSize.width,\r\n      height: popupSize.height,\r\n      top: 0,\r\n      left: 0,\r\n      right: popupSize.width,\r\n      bottom: popupSize.height,\r\n      x: 0,\r\n      y: 0,\r\n      toJSON: () => ({}),\r\n    } as DOMRect;\r\n\r\n    // 获取目标位置信息\r\n    const targetInfo = this.placementInfo;\r\n    let finalPlacement = targetInfo.placement;\r\n    let finalPosition = targetInfo.position;\r\n\r\n    if (this.autoAdjust) {\r\n      const optimal = this.calculateOptimalPlacement(\r\n        triggerRect,\r\n        popupRect,\r\n        viewportWidth,\r\n        viewportHeight\r\n      );\r\n      finalPlacement = optimal.placement;\r\n      finalPosition = optimal.position;\r\n    }\r\n\r\n    this.actualPlacement = finalPlacement;\r\n    this.actualPosition = finalPosition;\r\n\r\n    // 立即设置位置（即使元素还没完全渲染，也要先设置位置）\r\n    if (this.popupElement && this.popupElement.nativeElement) {\r\n      this.setPopupPosition(\r\n        triggerRect,\r\n        popupRect,\r\n        finalPlacement,\r\n        finalPosition,\r\n        popupSize\r\n      );\r\n      // 强制应用样式，确保位置立即生效\r\n      this.ref.detectChanges();\r\n    } else {\r\n      // 如果元素还不存在，延迟设置位置\r\n      setTimeout(() => {\r\n        if (this.popupElement && this.popupElement.nativeElement) {\r\n          this.setPopupPosition(\r\n            triggerRect,\r\n            popupRect,\r\n            finalPlacement,\r\n            finalPosition,\r\n            popupSize\r\n          );\r\n          this.ref.detectChanges();\r\n        }\r\n      }, 0);\r\n    }\r\n\r\n    this.ref.markForCheck();\r\n  }\r\n\r\n  updatePopupPosition() {\r\n    if (\r\n      !this.currentTriggerElement ||\r\n      !this.popupElement ||\r\n      !this.popupElement.nativeElement\r\n    ) {\r\n      return;\r\n    }\r\n\r\n    // 如果提供了固定尺寸，使用固定尺寸计算\r\n    if (this._width !== null && this._height !== null) {\r\n      this.calculateAndSetPosition();\r\n      return;\r\n    }\r\n\r\n    const triggerRect = this.currentTriggerElement.getBoundingClientRect();\r\n    const popupRect = this.popupElement.nativeElement.getBoundingClientRect();\r\n    const viewportWidth = window.innerWidth;\r\n    const viewportHeight = window.innerHeight;\r\n\r\n    // 获取目标位置信息\r\n    const targetInfo = this.placementInfo;\r\n    let finalPlacement = targetInfo.placement;\r\n    let finalPosition = targetInfo.position;\r\n\r\n    if (this.autoAdjust) {\r\n      const optimal = this.calculateOptimalPlacement(\r\n        triggerRect,\r\n        popupRect,\r\n        viewportWidth,\r\n        viewportHeight\r\n      );\r\n      finalPlacement = optimal.placement;\r\n      finalPosition = optimal.position;\r\n    }\r\n\r\n    this.actualPlacement = finalPlacement;\r\n    this.actualPosition = finalPosition;\r\n    this.setPopupPosition(\r\n      triggerRect,\r\n      popupRect,\r\n      finalPlacement,\r\n      finalPosition\r\n    );\r\n    this.ref.markForCheck();\r\n  }\r\n\r\n  calculateOptimalPlacement(\r\n    triggerRect: DOMRect | ClientRect,\r\n    popupRect: DOMRect | ClientRect,\r\n    viewportWidth: number,\r\n    viewportHeight: number\r\n  ): {\r\n    placement: \"top\" | \"left\" | \"right\" | \"bottom\";\r\n    position: \"start\" | \"center\" | \"end\";\r\n  } {\r\n    const targetInfo = this.placementInfo;\r\n    const preferred = targetInfo.placement;\r\n    const preferredPosition = targetInfo.position;\r\n\r\n    const space = {\r\n      top: triggerRect.top,\r\n      bottom: viewportHeight - triggerRect.bottom,\r\n      left: triggerRect.left,\r\n      right: viewportWidth - triggerRect.right,\r\n    };\r\n\r\n    const popupSize = {\r\n      width: popupRect.width || 300, // 默认宽度，如果还没有渲染\r\n      height: popupRect.height || 200, // 默认高度\r\n    };\r\n\r\n    // 检查首选位置是否有足够空间\r\n    const hasEnoughSpace = (placement: \"top\" | \"left\" | \"right\" | \"bottom\") => {\r\n      switch (placement) {\r\n        case \"top\":\r\n          return space.top >= popupSize.height + this.offset;\r\n        case \"bottom\":\r\n          return space.bottom >= popupSize.height + this.offset;\r\n        case \"left\":\r\n          return space.left >= popupSize.width + this.offset;\r\n        case \"right\":\r\n          return space.right >= popupSize.width + this.offset;\r\n      }\r\n    };\r\n\r\n    // 检查指定位置是否有足够空间\r\n    const checkPositionSpace = (\r\n      placement: \"top\" | \"left\" | \"right\" | \"bottom\",\r\n      position: \"start\" | \"center\" | \"end\"\r\n    ): boolean => {\r\n      if (placement === \"top\" || placement === \"bottom\") {\r\n        // 对于上下位置，检查水平对齐空间\r\n        const popupWidth = popupSize.width;\r\n        const spaceLeft = triggerRect.left;\r\n        const spaceRight = viewportWidth - triggerRect.right;\r\n\r\n        if (position === \"center\") {\r\n          return spaceLeft >= popupWidth / 2 && spaceRight >= popupWidth / 2;\r\n        } else if (position === \"start\") {\r\n          return spaceLeft >= popupWidth;\r\n        } else {\r\n          // end\r\n          return spaceRight >= popupWidth;\r\n        }\r\n      } else {\r\n        // 对于左右位置，检查垂直对齐空间\r\n        const popupHeight = popupSize.height;\r\n        const spaceTop = triggerRect.top;\r\n        const spaceBottom = viewportHeight - triggerRect.bottom;\r\n\r\n        if (position === \"center\") {\r\n          return spaceTop >= popupHeight / 2 && spaceBottom >= popupHeight / 2;\r\n        } else if (position === \"start\") {\r\n          return spaceTop >= popupHeight;\r\n        } else {\r\n          // end\r\n          return spaceBottom >= popupHeight;\r\n        }\r\n      }\r\n    };\r\n\r\n    // 计算位置是否合适（考虑水平/垂直对齐）\r\n    const calculatePosition = (\r\n      placement: \"top\" | \"left\" | \"right\" | \"bottom\"\r\n    ): \"start\" | \"center\" | \"end\" => {\r\n      if (placement === \"top\" || placement === \"bottom\") {\r\n        // 对于上下位置，检查水平对齐\r\n        const popupWidth = popupSize.width;\r\n        const spaceLeft = triggerRect.left;\r\n        const spaceRight = viewportWidth - triggerRect.right;\r\n\r\n        // 优先尝试使用用户指定的position\r\n        if (checkPositionSpace(placement, preferredPosition)) {\r\n          return preferredPosition;\r\n        }\r\n\r\n        // 如果中心对齐有足够空间，优先使用中心\r\n        if (spaceLeft >= popupWidth / 2 && spaceRight >= popupWidth / 2) {\r\n          return \"center\";\r\n        }\r\n        // 如果左侧空间不足，使用右对齐\r\n        if (spaceLeft < popupWidth / 2) {\r\n          return \"end\";\r\n        }\r\n        // 如果右侧空间不足，使用左对齐\r\n        if (spaceRight < popupWidth / 2) {\r\n          return \"start\";\r\n        }\r\n        return preferredPosition;\r\n      } else {\r\n        // 对于左右位置，检查垂直对齐\r\n        const popupHeight = popupSize.height;\r\n        const spaceTop = triggerRect.top;\r\n        const spaceBottom = viewportHeight - triggerRect.bottom;\r\n\r\n        // 优先尝试使用用户指定的position\r\n        if (checkPositionSpace(placement, preferredPosition)) {\r\n          return preferredPosition;\r\n        }\r\n\r\n        // 如果中心对齐有足够空间，优先使用中心\r\n        if (spaceTop >= popupHeight / 2 && spaceBottom >= popupHeight / 2) {\r\n          return \"center\";\r\n        }\r\n        // 如果上方空间不足，使用下对齐\r\n        if (spaceTop < popupHeight / 2) {\r\n          return \"end\";\r\n        }\r\n        // 如果下方空间不足，使用上对齐\r\n        if (spaceBottom < popupHeight / 2) {\r\n          return \"start\";\r\n        }\r\n        return preferredPosition;\r\n      }\r\n    };\r\n\r\n    // 如果首选位置有足够空间，优先使用用户指定的position\r\n    if (hasEnoughSpace(preferred)) {\r\n      // 检查用户指定的position是否有足够空间\r\n      if (checkPositionSpace(preferred, preferredPosition)) {\r\n        return {\r\n          placement: preferred,\r\n          position: preferredPosition,\r\n        };\r\n      }\r\n      // 如果指定的position空间不足，自动计算最佳position\r\n      return {\r\n        placement: preferred,\r\n        position: calculatePosition(preferred),\r\n      };\r\n    }\r\n\r\n    // 根据首选位置选择备选位置\r\n    const alternatives: {\r\n      [key: string]: (\"top\" | \"left\" | \"right\" | \"bottom\")[];\r\n    } = {\r\n      bottom: [\"top\", \"right\", \"left\"],\r\n      top: [\"bottom\", \"right\", \"left\"],\r\n      right: [\"left\", \"bottom\", \"top\"],\r\n      left: [\"right\", \"bottom\", \"top\"],\r\n    };\r\n\r\n    const altList = alternatives[preferred] || [];\r\n\r\n    // 尝试找到第一个有足够空间的备选位置\r\n    for (const alt of altList) {\r\n      if (hasEnoughSpace(alt)) {\r\n        return {\r\n          placement: alt,\r\n          position: calculatePosition(alt),\r\n        };\r\n      }\r\n    }\r\n\r\n    // 如果都没有足够空间，选择空间最大的位置\r\n    const maxSpace = Math.max(space.top, space.bottom, space.left, space.right);\r\n    let bestPlacement: \"top\" | \"left\" | \"right\" | \"bottom\";\r\n    if (maxSpace === space.top) bestPlacement = \"top\";\r\n    else if (maxSpace === space.bottom) bestPlacement = \"bottom\";\r\n    else if (maxSpace === space.left) bestPlacement = \"left\";\r\n    else bestPlacement = \"right\";\r\n\r\n    return {\r\n      placement: bestPlacement,\r\n      position: calculatePosition(bestPlacement),\r\n    };\r\n  }\r\n\r\n  // 计算位置坐标（纯计算，不依赖DOM，包含边界约束）\r\n  private calculatePositionCoordinates(\r\n    triggerRect: DOMRect | ClientRect,\r\n    placement: \"top\" | \"left\" | \"right\" | \"bottom\",\r\n    position: \"start\" | \"center\" | \"end\",\r\n    popupSize: { width: number; height: number }\r\n  ): { top: string; left: string; translateX: string; translateY: string } {\r\n    const triggerWidth = triggerRect.width;\r\n    const triggerHeight = triggerRect.height;\r\n    const viewportWidth = window.innerWidth;\r\n    const viewportHeight = window.innerHeight;\r\n    const scrollX = window.scrollX;\r\n    const scrollY = window.scrollY;\r\n\r\n    let top = 0;\r\n    let left = 0;\r\n    let translateX = \"0\";\r\n    let translateY = \"0\";\r\n\r\n    switch (placement) {\r\n      case \"top\":\r\n        top = triggerRect.top + scrollY - this.offset;\r\n        translateY = \"-100%\";\r\n        if (position === \"start\") {\r\n          left = triggerRect.left + scrollX;\r\n        } else if (position === \"center\") {\r\n          left = triggerRect.left + scrollX + triggerWidth / 2;\r\n          translateX = \"-50%\";\r\n        } else {\r\n          left = triggerRect.left + scrollX + triggerWidth;\r\n          translateX = \"-100%\";\r\n        }\r\n        // 边界约束：确保弹窗不超出视口\r\n        if (top - popupSize.height < scrollY) {\r\n          top = scrollY + 8; // 距离顶部至少8px\r\n        }\r\n        if (left < scrollX) {\r\n          left = scrollX + 8;\r\n          translateX = \"0\";\r\n        } else if (left + popupSize.width > scrollX + viewportWidth) {\r\n          left = scrollX + viewportWidth - popupSize.width - 8;\r\n          translateX = \"0\";\r\n        }\r\n        break;\r\n\r\n      case \"bottom\":\r\n        top = triggerRect.bottom + scrollY + this.offset;\r\n        if (position === \"start\") {\r\n          left = triggerRect.left + scrollX;\r\n        } else if (position === \"center\") {\r\n          left = triggerRect.left + scrollX + triggerWidth / 2;\r\n          translateX = \"-50%\";\r\n        } else {\r\n          left = triggerRect.left + scrollX + triggerWidth;\r\n          translateX = \"-100%\";\r\n        }\r\n        // 边界约束：确保弹窗不超出视口\r\n        if (top + popupSize.height > scrollY + viewportHeight) {\r\n          top = scrollY + viewportHeight - popupSize.height - 8; // 距离底部至少8px\r\n        }\r\n        if (left < scrollX) {\r\n          left = scrollX + 8;\r\n          translateX = \"0\";\r\n        } else if (left + popupSize.width > scrollX + viewportWidth) {\r\n          left = scrollX + viewportWidth - popupSize.width - 8;\r\n          translateX = \"0\";\r\n        }\r\n        break;\r\n\r\n      case \"left\":\r\n        left = triggerRect.left + scrollX - this.offset;\r\n        translateX = \"-100%\";\r\n        if (position === \"start\") {\r\n          top = triggerRect.top + scrollY;\r\n        } else if (position === \"center\") {\r\n          top = triggerRect.top + scrollY + triggerHeight / 2;\r\n          translateY = \"-50%\";\r\n        } else {\r\n          top = triggerRect.top + scrollY + triggerHeight;\r\n          translateY = \"-100%\";\r\n        }\r\n        // 边界约束：确保弹窗不超出视口\r\n        if (left - popupSize.width < scrollX) {\r\n          left = scrollX + 8; // 距离左侧至少8px\r\n        }\r\n        if (top < scrollY) {\r\n          top = scrollY + 8;\r\n          translateY = \"0\";\r\n        } else if (top + popupSize.height > scrollY + viewportHeight) {\r\n          top = scrollY + viewportHeight - popupSize.height - 8;\r\n          translateY = \"0\";\r\n        }\r\n        break;\r\n\r\n      case \"right\":\r\n        left = triggerRect.right + scrollX + this.offset;\r\n        if (position === \"start\") {\r\n          top = triggerRect.top + scrollY;\r\n        } else if (position === \"center\") {\r\n          top = triggerRect.top + scrollY + triggerHeight / 2;\r\n          translateY = \"-50%\";\r\n        } else {\r\n          top = triggerRect.top + scrollY + triggerHeight;\r\n          translateY = \"-100%\";\r\n        }\r\n        // 边界约束：确保弹窗不超出视口\r\n        if (left + popupSize.width > scrollX + viewportWidth) {\r\n          left = scrollX + viewportWidth - popupSize.width - 8; // 距离右侧至少8px\r\n        }\r\n        if (top < scrollY) {\r\n          top = scrollY + 8;\r\n          translateY = \"0\";\r\n        } else if (top + popupSize.height > scrollY + viewportHeight) {\r\n          top = scrollY + viewportHeight - popupSize.height - 8;\r\n          translateY = \"0\";\r\n        }\r\n        break;\r\n    }\r\n\r\n    return {\r\n      top: top + \"px\",\r\n      left: left + \"px\",\r\n      translateX,\r\n      translateY,\r\n    };\r\n  }\r\n\r\n  setPopupPosition(\r\n    triggerRect: DOMRect | ClientRect,\r\n    popupRect: DOMRect | ClientRect,\r\n    placement: \"top\" | \"left\" | \"right\" | \"bottom\",\r\n    position: \"start\" | \"center\" | \"end\",\r\n    popupSize?: { width: number; height: number }\r\n  ) {\r\n    const popupEl =\r\n      this.popupElement && this.popupElement.nativeElement\r\n        ? this.popupElement.nativeElement\r\n        : null;\r\n    if (!popupEl) {\r\n      return; // 如果元素还不存在，无法设置位置\r\n    }\r\n\r\n    // 优先使用传入的固定尺寸，否则从DOM获取\r\n    const popupWidth =\r\n      popupSize && popupSize.width !== null && popupSize.width !== undefined\r\n        ? popupSize.width\r\n        : popupRect.width || popupEl.offsetWidth || 300;\r\n    const popupHeight =\r\n      popupSize && popupSize.height !== null && popupSize.height !== undefined\r\n        ? popupSize.height\r\n        : popupRect.height || popupEl.offsetHeight || 200;\r\n\r\n    const coordinates = this.calculatePositionCoordinates(\r\n      triggerRect,\r\n      placement,\r\n      position,\r\n      { width: popupWidth, height: popupHeight }\r\n    );\r\n\r\n    popupEl.style.top = coordinates.top;\r\n    popupEl.style.left = coordinates.left;\r\n    popupEl.style.transform = `translate(${coordinates.translateX}, ${coordinates.translateY})`;\r\n  }\r\n\r\n  @HostListener(\"window:resize\")\r\n  onWindowResize() {\r\n    if (this.opened === \"yes\") {\r\n      this.updatePopupPosition();\r\n    }\r\n  }\r\n}\r\n"]}
|