raise-common-lib-new 0.0.62 → 0.0.64

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (223) hide show
  1. package/esm2022/lib/actions/toolbar/index.component.mjs +4 -3
  2. package/esm2022/lib/actions/toolbar-item/index.component.mjs +2 -2
  3. package/esm2022/lib/common-grid/grid-action/grid-action-item/grid-action-item.component.mjs +1 -1
  4. package/esm2022/lib/common-grid/grid-action/grid-action.component.mjs +3 -2
  5. package/esm2022/lib/common-grid/index.component.mjs +6 -3
  6. package/esm2022/lib/dashboard/api.mjs +33 -0
  7. package/esm2022/lib/dashboard/bar-charts/bar-charts.component.mjs +430 -0
  8. package/esm2022/lib/dashboard/bar-charts/utils.mjs +18 -0
  9. package/esm2022/lib/dashboard/dashboard-properties/data-mart-new/data-mart-new.component.mjs +442 -0
  10. package/esm2022/lib/dashboard/dashboard.service.mjs +295 -0
  11. package/esm2022/lib/dashboard/dashboardPorlets.service.mjs +2169 -0
  12. package/esm2022/lib/dashboard/dialog-group/download/index.component.mjs +212 -0
  13. package/esm2022/lib/dashboard/dialog-group/empty-icon-prompt/empty-icon-prompt.component.mjs +36 -0
  14. package/esm2022/lib/dashboard/gadget-group/gadget-pivot/gadget-pivot.component.mjs +3211 -0
  15. package/esm2022/lib/dashboard/gadget-group/gadget-pivot-chart/gadget-pivot.component.mjs +3117 -0
  16. package/esm2022/lib/dashboard/gadget-group/gadget-table/gadget-table.component.mjs +1099 -0
  17. package/esm2022/lib/dashboard/gadget-group/gadget-transpose/gadget-transpose.component.mjs +583 -0
  18. package/esm2022/lib/dashboard/pane-group-new.component.mjs +2031 -0
  19. package/esm2022/lib/dashboard/sidebar-iconlist/field-filter/field-filter.component.mjs +637 -0
  20. package/esm2022/lib/dashboard/sidebar-iconlist/field-format/field-format.component.mjs +753 -0
  21. package/esm2022/lib/dashboard/sidebar-iconlist/portlet-type-new/portlet-type-new.component.mjs +216 -0
  22. package/esm2022/lib/dashboard/sidebar-iconlist/sidebar-iconlist-new.component.mjs +1239 -0
  23. package/esm2022/lib/dialog/common-delete-dialog/index.component.mjs +1 -1
  24. package/esm2022/lib/dialog/common-dialog/index.component.mjs +16 -5
  25. package/esm2022/lib/float-box/index.component.mjs +18 -4
  26. package/esm2022/lib/form/checkbox-group/index.component.mjs +35 -5
  27. package/esm2022/lib/form/drawer-form/drawer-form.component.mjs +177 -76
  28. package/esm2022/lib/form/richtexteditor/index.component.mjs +7 -9
  29. package/esm2022/lib/layout/drawer/index.component.mjs +2 -2
  30. package/esm2022/lib/layout/page-list/index.component.mjs +9 -3
  31. package/esm2022/lib/layout/page-tab/index.component.mjs +5 -3
  32. package/esm2022/lib/layout/rs-stepper/constants.mjs +2 -0
  33. package/esm2022/lib/layout/rs-stepper/index.component.mjs +148 -0
  34. package/esm2022/lib/raise-common-lib.module.mjs +114 -7
  35. package/esm2022/lib/service/InjectionToken.mjs +5 -0
  36. package/esm2022/lib/service/keep-alive.service.mjs +2 -2
  37. package/esm2022/lib/smart-popup/index.component.mjs +1084 -0
  38. package/esm2022/public-api.mjs +19 -1
  39. package/fesm2022/raise-common-lib-new.mjs +18042 -227
  40. package/fesm2022/raise-common-lib-new.mjs.map +1 -1
  41. package/lib/common-grid/index.component.d.ts +2 -1
  42. package/lib/dashboard/api.d.ts +24 -0
  43. package/lib/dashboard/bar-charts/bar-charts.component.d.ts +63 -0
  44. package/lib/dashboard/bar-charts/utils.d.ts +1 -0
  45. package/lib/dashboard/dashboard-properties/data-mart-new/data-mart-new.component.d.ts +52 -0
  46. package/lib/dashboard/dashboard.service.d.ts +152 -0
  47. package/lib/dashboard/dashboardPorlets.service.d.ts +214 -0
  48. package/lib/dashboard/dialog-group/download/index.component.d.ts +36 -0
  49. package/lib/dashboard/dialog-group/empty-icon-prompt/empty-icon-prompt.component.d.ts +15 -0
  50. package/lib/dashboard/gadget-group/gadget-pivot/gadget-pivot.component.d.ts +128 -0
  51. package/lib/dashboard/gadget-group/gadget-pivot-chart/gadget-pivot.component.d.ts +131 -0
  52. package/lib/dashboard/gadget-group/gadget-table/gadget-table.component.d.ts +77 -0
  53. package/lib/dashboard/gadget-group/gadget-transpose/gadget-transpose.component.d.ts +47 -0
  54. package/lib/dashboard/pane-group-new.component.d.ts +169 -0
  55. package/lib/dashboard/sidebar-iconlist/field-filter/field-filter.component.d.ts +56 -0
  56. package/lib/dashboard/sidebar-iconlist/field-format/field-format.component.d.ts +62 -0
  57. package/lib/dashboard/sidebar-iconlist/portlet-type-new/portlet-type-new.component.d.ts +30 -0
  58. package/lib/dashboard/sidebar-iconlist/sidebar-iconlist-new.component.d.ts +150 -0
  59. package/lib/dialog/common-dialog/index.component.d.ts +3 -1
  60. package/lib/float-box/index.component.d.ts +4 -3
  61. package/lib/form/checkbox-group/index.component.d.ts +5 -0
  62. package/lib/form/drawer-form/drawer-form.component.d.ts +21 -8
  63. package/lib/form/richtexteditor/index.component.d.ts +1 -3
  64. package/lib/layout/page-list/index.component.d.ts +3 -1
  65. package/lib/layout/page-tab/index.component.d.ts +1 -0
  66. package/lib/layout/rs-stepper/constants.d.ts +5 -0
  67. package/lib/layout/rs-stepper/index.component.d.ts +31 -0
  68. package/lib/raise-common-lib.module.d.ts +61 -40
  69. package/lib/service/InjectionToken.d.ts +4 -0
  70. package/lib/smart-popup/index.component.d.ts +60 -0
  71. package/package.json +1 -1
  72. package/public-api.d.ts +18 -0
  73. package/src/assets/img/dashboard_icon/AddPerson.svg +4 -0
  74. package/src/assets/img/dashboard_icon/Angle-double-left.svg +12 -0
  75. package/src/assets/img/dashboard_icon/Angle-double-right.svg +12 -0
  76. package/src/assets/img/dashboard_icon/Arrow_collapse.svg +3 -0
  77. package/src/assets/img/dashboard_icon/Arrow_expand.svg +3 -0
  78. package/src/assets/img/dashboard_icon/Close.svg +4 -0
  79. package/src/assets/img/dashboard_icon/Edit_label.svg +3 -0
  80. package/src/assets/img/dashboard_icon/RemoveMini.svg +3 -0
  81. package/src/assets/img/dashboard_icon/ShareTo.svg +3 -0
  82. package/src/assets/img/dashboard_icon/add-dashboard.svg +4 -0
  83. package/src/assets/img/dashboard_icon/add-tab.svg +4 -0
  84. package/src/assets/img/dashboard_icon/angle-left.svg +12 -0
  85. package/src/assets/img/dashboard_icon/angle-right.svg +12 -0
  86. package/src/assets/img/dashboard_icon/area-white.svg +3 -0
  87. package/src/assets/img/dashboard_icon/area.svg +3 -0
  88. package/src/assets/img/dashboard_icon/bar-white.svg +14 -0
  89. package/src/assets/img/dashboard_icon/bar.svg +14 -0
  90. package/src/assets/img/dashboard_icon/basic-icon-filter.svg +14 -0
  91. package/src/assets/img/dashboard_icon/chart-combined-active.svg +7 -0
  92. package/src/assets/img/dashboard_icon/chart-combined.svg +7 -0
  93. package/src/assets/img/dashboard_icon/column-white.svg +14 -0
  94. package/src/assets/img/dashboard_icon/column.svg +26 -0
  95. package/src/assets/img/dashboard_icon/customize.png +0 -0
  96. package/src/assets/img/dashboard_icon/dashboard-description.svg +18 -0
  97. package/src/assets/img/dashboard_icon/dashboard-download.svg +18 -0
  98. package/src/assets/img/dashboard_icon/dashboard-drag.svg +28 -0
  99. package/src/assets/img/dashboard_icon/dashboard-duplicate.svg +24 -0
  100. package/src/assets/img/dashboard_icon/dashboard-icon.svg +12 -0
  101. package/src/assets/img/dashboard_icon/dashboard-properties.svg +18 -0
  102. package/src/assets/img/dashboard_icon/dashboard-rename.svg +18 -0
  103. package/src/assets/img/dashboard_icon/dashboard-share.svg +20 -0
  104. package/src/assets/img/dashboard_icon/dashboard-tab-delete.svg +4 -0
  105. package/src/assets/img/dashboard_icon/dashboard_Share.svg +3 -0
  106. package/src/assets/img/dashboard_icon/dashboard_ToolsHide.svg +4 -0
  107. package/src/assets/img/dashboard_icon/datamart_.svg +5 -0
  108. package/src/assets/img/dashboard_icon/datamart_Admin.svg +3 -0
  109. package/src/assets/img/dashboard_icon/datamart_Asset Portfolio.svg +4 -0
  110. package/src/assets/img/dashboard_icon/datamart_Bank.svg +3 -0
  111. package/src/assets/img/dashboard_icon/datamart_CRM.svg +6 -0
  112. package/src/assets/img/dashboard_icon/datamart_Company.svg +6 -0
  113. package/src/assets/img/dashboard_icon/datamart_Compliance.svg +3 -0
  114. package/src/assets/img/dashboard_icon/datamart_Contact.svg +3 -0
  115. package/src/assets/img/dashboard_icon/datamart_DASHBOARD.svg +4 -0
  116. package/src/assets/img/dashboard_icon/datamart_DOCX.svg +4 -0
  117. package/src/assets/img/dashboard_icon/datamart_ESG.svg +4 -0
  118. package/src/assets/img/dashboard_icon/datamart_Financials - Company.svg +3 -0
  119. package/src/assets/img/dashboard_icon/datamart_Fund.svg +4 -0
  120. package/src/assets/img/dashboard_icon/datamart_FundFinancial.svg +4 -0
  121. package/src/assets/img/dashboard_icon/datamart_HTML.svg +4 -0
  122. package/src/assets/img/dashboard_icon/datamart_Logs.svg +8 -0
  123. package/src/assets/img/dashboard_icon/datamart_PDF.svg +5 -0
  124. package/src/assets/img/dashboard_icon/datamart_Project.svg +3 -0
  125. package/src/assets/img/dashboard_icon/datamart_Templates.svg +7 -0
  126. package/src/assets/img/dashboard_icon/datamart_Track Change.svg +3 -0
  127. package/src/assets/img/dashboard_icon/datamart_VirtualGroup.svg +5 -0
  128. package/src/assets/img/dashboard_icon/datamart_XLS.svg +4 -0
  129. package/src/assets/img/dashboard_icon/desktop_selected.svg +10 -0
  130. package/src/assets/img/dashboard_icon/desktop_unselected.svg +10 -0
  131. package/src/assets/img/dashboard_icon/favourite-grey.svg +3 -0
  132. package/src/assets/img/dashboard_icon/favourite-yellow.svg +3 -0
  133. package/src/assets/img/dashboard_icon/forms-checkbox-square-tick.svg +3 -0
  134. package/src/assets/img/dashboard_icon/ftable-white.svg +24 -0
  135. package/src/assets/img/dashboard_icon/ftable.svg +24 -0
  136. package/src/assets/img/dashboard_icon/gadget-basic-arrow-down.svg +14 -0
  137. package/src/assets/img/dashboard_icon/gadget-basic-format.svg +21 -0
  138. package/src/assets/img/dashboard_icon/gadget-basic-sub-total.svg +18 -0
  139. package/src/assets/img/dashboard_icon/gadget-basic-total-1.svg +9 -0
  140. package/src/assets/img/dashboard_icon/gadget-basic-total.svg +16 -0
  141. package/src/assets/img/dashboard_icon/gadget-columns.svg +12 -0
  142. package/src/assets/img/dashboard_icon/gadget-delete.svg +18 -0
  143. package/src/assets/img/dashboard_icon/gadget-download.svg +18 -0
  144. package/src/assets/img/dashboard_icon/gadget-duplicate.svg +24 -0
  145. package/src/assets/img/dashboard_icon/gadget-edit.svg +18 -0
  146. package/src/assets/img/dashboard_icon/gadget-filters-light.svg +9 -0
  147. package/src/assets/img/dashboard_icon/gadget-filters.svg +9 -0
  148. package/src/assets/img/dashboard_icon/gadget-format-0.svg +8 -0
  149. package/src/assets/img/dashboard_icon/gadget-format-1.svg +8 -0
  150. package/src/assets/img/dashboard_icon/gadget-format.svg +19 -0
  151. package/src/assets/img/dashboard_icon/gadget-fullscreen.svg +30 -0
  152. package/src/assets/img/dashboard_icon/gadget-settings.svg +18 -0
  153. package/src/assets/img/dashboard_icon/gadget-sub-total-0.svg +18 -0
  154. package/src/assets/img/dashboard_icon/gadget-sub-total-1.svg +18 -0
  155. package/src/assets/img/dashboard_icon/gadget-thumbnail.svg +12 -0
  156. package/src/assets/img/dashboard_icon/gadget-type-light.svg +15 -0
  157. package/src/assets/img/dashboard_icon/gadget-type.svg +16 -0
  158. package/src/assets/img/dashboard_icon/geo-white.svg +14 -0
  159. package/src/assets/img/dashboard_icon/geo.svg +14 -0
  160. package/src/assets/img/dashboard_icon/historic-IRR-white.svg +9 -0
  161. package/src/assets/img/dashboard_icon/historic-IRR.svg +5 -0
  162. package/src/assets/img/dashboard_icon/information-1.png +0 -0
  163. package/src/assets/img/dashboard_icon/information-grey.svg +4 -0
  164. package/src/assets/img/dashboard_icon/menu-change-group.svg +3 -0
  165. package/src/assets/img/dashboard_icon/menu-close.svg +20 -0
  166. package/src/assets/img/dashboard_icon/menu-delete.svg +18 -0
  167. package/src/assets/img/dashboard_icon/menu-description.svg +19 -0
  168. package/src/assets/img/dashboard_icon/menu-download.svg +24 -0
  169. package/src/assets/img/dashboard_icon/menu-drag.svg +28 -0
  170. package/src/assets/img/dashboard_icon/menu-duplicate.svg +26 -0
  171. package/src/assets/img/dashboard_icon/menu-properties.svg +18 -0
  172. package/src/assets/img/dashboard_icon/menu-rename.svg +23 -0
  173. package/src/assets/img/dashboard_icon/menu-share.svg +20 -0
  174. package/src/assets/img/dashboard_icon/menu-toolbar.svg +10 -0
  175. package/src/assets/img/dashboard_icon/mobile_selected.svg +10 -0
  176. package/src/assets/img/dashboard_icon/mobile_unselected.svg +10 -0
  177. package/src/assets/img/dashboard_icon/more_Save.svg +3 -0
  178. package/src/assets/img/dashboard_icon/more_View.svg +3 -0
  179. package/src/assets/img/dashboard_icon/multi-series-white.svg +20 -0
  180. package/src/assets/img/dashboard_icon/multi-series.svg +6 -0
  181. package/src/assets/img/dashboard_icon/paint-bucket.svg +6 -0
  182. package/src/assets/img/dashboard_icon/pie-white.svg +16 -0
  183. package/src/assets/img/dashboard_icon/pie.svg +16 -0
  184. package/src/assets/img/dashboard_icon/pivot-chart-active.svg +8 -0
  185. package/src/assets/img/dashboard_icon/pivot-chart.svg +8 -0
  186. package/src/assets/img/dashboard_icon/pivot-white.svg +8 -0
  187. package/src/assets/img/dashboard_icon/pivot.svg +8 -0
  188. package/src/assets/img/dashboard_icon/pivot_column.svg +9 -0
  189. package/src/assets/img/dashboard_icon/pivot_column_active.svg +9 -0
  190. package/src/assets/img/dashboard_icon/pivot_line.svg +12 -0
  191. package/src/assets/img/dashboard_icon/pivot_line_active.svg +12 -0
  192. package/src/assets/img/dashboard_icon/print-A2.svg +3 -0
  193. package/src/assets/img/dashboard_icon/radar-white.svg +19 -0
  194. package/src/assets/img/dashboard_icon/radar.svg +19 -0
  195. package/src/assets/img/dashboard_icon/saveAs.svg +3 -0
  196. package/src/assets/img/dashboard_icon/search_input.svg +4 -0
  197. package/src/assets/img/dashboard_icon/table-3-white.svg +16 -0
  198. package/src/assets/img/dashboard_icon/table-3.svg +16 -0
  199. package/src/assets/img/dashboard_icon/table-filter-off.svg +5 -0
  200. package/src/assets/img/dashboard_icon/table-filter-on.svg +5 -0
  201. package/src/assets/img/dashboard_icon/tag_dashboard_selected.svg +3 -0
  202. package/src/assets/img/dashboard_icon/tag_dashboard_unselected.svg +3 -0
  203. package/src/assets/img/dashboard_icon/tag_share_selected.svg +3 -0
  204. package/src/assets/img/dashboard_icon/tag_share_unselected.svg +3 -0
  205. package/src/assets/img/dashboard_icon/tick.svg +26 -0
  206. package/src/assets/img/dashboard_icon/toolbar_AddDashboard.svg +5 -0
  207. package/src/assets/img/dashboard_icon/toolbar_AddGroup.svg +3 -0
  208. package/src/assets/img/dashboard_icon/toolbar_AddReport.svg +3 -0
  209. package/src/assets/img/dashboard_icon/toolbar_CollapseAll.svg +3 -0
  210. package/src/assets/img/dashboard_icon/toolbar_Customize.svg +3 -0
  211. package/src/assets/img/dashboard_icon/toolbar_DeleteDark.svg +3 -0
  212. package/src/assets/img/dashboard_icon/toolbar_Duplicate.svg +3 -0
  213. package/src/assets/img/dashboard_icon/toolbar_Edit.svg +3 -0
  214. package/src/assets/img/dashboard_icon/toolbar_ExpandAll.svg +3 -0
  215. package/src/assets/img/dashboard_icon/toolbar_Search.svg +3 -0
  216. package/src/assets/img/dashboard_icon/topbar-menu.svg +33 -0
  217. package/src/assets/img/dashboard_icon/topbar-refresh.svg +17 -0
  218. package/src/assets/img/dashboard_icon/topbar-tool-off.svg +12 -0
  219. package/src/assets/img/dashboard_icon/topbar-tool-on.svg +12 -0
  220. package/src/assets/img/dashboard_icon/trade-up-white.svg +14 -0
  221. package/src/assets/img/dashboard_icon/trade-up.svg +14 -0
  222. package/src/assets/img/dashboard_icon/transpose-white.svg +16 -0
  223. package/src/assets/img/dashboard_icon/transpose.svg +16 -0
@@ -0,0 +1,1084 @@
1
+ import { Component, EventEmitter, Input, Output, ViewChild, HostListener, } from "@angular/core";
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/common";
4
+ export class SmartPopupComponent {
5
+ ref;
6
+ _placement = "BottomLeft";
7
+ _offset = 8;
8
+ _autoAdjust = "yes";
9
+ triggerElementRef = null;
10
+ loading = false;
11
+ _open = false;
12
+ _width = null;
13
+ _height = null;
14
+ // 解析placement格式
15
+ get placementInfo() {
16
+ const placementMap = {
17
+ TopLeft: { placement: "top", position: "start" },
18
+ TopCenter: { placement: "top", position: "center" },
19
+ TopRight: { placement: "top", position: "end" },
20
+ BottomLeft: { placement: "bottom", position: "start" },
21
+ BottomCenter: { placement: "bottom", position: "center" },
22
+ BottomRight: { placement: "bottom", position: "end" },
23
+ LeftTop: { placement: "left", position: "start" },
24
+ LeftCenter: { placement: "left", position: "center" },
25
+ LeftBottom: { placement: "left", position: "end" },
26
+ RightTop: { placement: "right", position: "start" },
27
+ RightCenter: { placement: "right", position: "center" },
28
+ RightBottom: { placement: "right", position: "end" },
29
+ };
30
+ return (placementMap[this._placement] || {
31
+ placement: "bottom",
32
+ position: "start",
33
+ });
34
+ }
35
+ get offset() {
36
+ return this._offset || 8;
37
+ }
38
+ get autoAdjust() {
39
+ return this._autoAdjust === "yes";
40
+ }
41
+ openChange = new EventEmitter();
42
+ contentLoad = new EventEmitter();
43
+ internalTriggerElement;
44
+ popupElement;
45
+ currentTriggerElement = null;
46
+ constructor(ref) {
47
+ this.ref = ref;
48
+ }
49
+ ngAfterViewInit() {
50
+ this.fixedContainerEl = this.getFixedContainer();
51
+ this.fixedContainerEl.append(this.popupElement.nativeElement);
52
+ this.updateTriggerElement();
53
+ // 如果提供了固定尺寸,立即应用
54
+ if (this._width !== null &&
55
+ this._height !== null &&
56
+ this.popupElement &&
57
+ this.popupElement.nativeElement) {
58
+ const popupEl = this.popupElement.nativeElement;
59
+ popupEl.style.width = this._width + "px";
60
+ popupEl.style.height = this._height + "px";
61
+ popupEl.style.minWidth = this._width + "px";
62
+ popupEl.style.maxWidth = this._width + "px";
63
+ popupEl.style.minHeight = this._height + "px";
64
+ popupEl.style.maxHeight = this._height + "px";
65
+ }
66
+ // 如果外部传入 open 属性为 true,自动打开
67
+ if (this._open) {
68
+ setTimeout(() => this.open(), 0);
69
+ }
70
+ }
71
+ ngOnChanges(changes) {
72
+ // 处理固定尺寸变化
73
+ if ((changes._width || changes._height) &&
74
+ this.popupElement &&
75
+ this.popupElement.nativeElement) {
76
+ const popupEl = this.popupElement.nativeElement;
77
+ if (this._width !== null) {
78
+ popupEl.style.width = this._width + "px";
79
+ popupEl.style.minWidth = this._width + "px";
80
+ popupEl.style.maxWidth = this._width + "px";
81
+ }
82
+ if (this._height !== null) {
83
+ popupEl.style.height = this._height + "px";
84
+ popupEl.style.minHeight = this._height + "px";
85
+ popupEl.style.maxHeight = this._height + "px";
86
+ }
87
+ }
88
+ // 先处理触发元素变化(确保在打开前更新触发元素)
89
+ if (changes.triggerElementRef) {
90
+ const previousTrigger = this.currentTriggerElement;
91
+ this.updateTriggerElement();
92
+ // 如果弹窗已经打开,且触发元素发生了变化,立即更新位置
93
+ if (this.opened === "yes" &&
94
+ previousTrigger !== this.currentTriggerElement &&
95
+ this.currentTriggerElement) {
96
+ // 如果提供了固定尺寸,直接重新计算位置(避免位置跳动)
97
+ if (this._width !== null && this._height !== null) {
98
+ // 先隐藏弹窗(positioning = yes)
99
+ this.positioning = "yes";
100
+ this.ref.markForCheck();
101
+ this.ref.detectChanges(); // 立即应用隐藏状态
102
+ requestAnimationFrame(() => {
103
+ if (this.currentTriggerElement) {
104
+ const triggerRect = this.currentTriggerElement.getBoundingClientRect();
105
+ const popupSize = {
106
+ width: this._width,
107
+ height: this._height,
108
+ };
109
+ const targetInfo = this.placementInfo;
110
+ let finalPlacement = targetInfo.placement;
111
+ const finalPosition = targetInfo.position;
112
+ if (this.autoAdjust) {
113
+ const popupRect = {
114
+ width: popupSize.width,
115
+ height: popupSize.height,
116
+ top: 0,
117
+ left: 0,
118
+ right: popupSize.width,
119
+ bottom: popupSize.height,
120
+ x: 0,
121
+ y: 0,
122
+ toJSON: () => ({}),
123
+ };
124
+ const optimal = this.calculateOptimalPlacement(triggerRect, popupRect, window.innerWidth, window.innerHeight);
125
+ finalPlacement = optimal.placement;
126
+ }
127
+ this.actualPlacement = finalPlacement;
128
+ this.actualPosition = finalPosition;
129
+ const position = this.calculatePositionCoordinates(triggerRect, finalPlacement, finalPosition, popupSize);
130
+ if (this.popupElement && this.popupElement.nativeElement) {
131
+ const popupEl = this.popupElement.nativeElement;
132
+ popupEl.style.top = position.top;
133
+ popupEl.style.left = position.left;
134
+ popupEl.style.transform = `translate(${position.translateX}, ${position.translateY})`;
135
+ popupEl.offsetHeight; // 强制同步应用样式
136
+ // 位置设置完成后,再显示弹窗
137
+ requestAnimationFrame(() => {
138
+ this.positioning = "no";
139
+ this.ref.markForCheck();
140
+ });
141
+ }
142
+ }
143
+ });
144
+ }
145
+ else {
146
+ // 使用新的打开方式更新位置
147
+ this.positioning = "yes";
148
+ this.ref.markForCheck();
149
+ requestAnimationFrame(() => {
150
+ requestAnimationFrame(() => {
151
+ this.updatePopupPosition();
152
+ this.positioning = "no";
153
+ this.ref.markForCheck();
154
+ });
155
+ });
156
+ }
157
+ }
158
+ }
159
+ // 处理 open 变化(确保在触发元素更新后处理)
160
+ if (changes._open) {
161
+ if (this._open && this.opened === "no") {
162
+ // 确保触发元素已更新
163
+ this.updateTriggerElement();
164
+ // 使用 setTimeout 确保所有变更都已处理
165
+ setTimeout(() => {
166
+ // 再次确保触发元素已更新
167
+ this.updateTriggerElement();
168
+ this.open();
169
+ }, 0);
170
+ }
171
+ else if (!this._open && this.opened === "yes") {
172
+ this.close();
173
+ }
174
+ }
175
+ // 当加载完成时,更新位置(如果提供了固定尺寸,不需要更新位置)
176
+ if (changes.loading &&
177
+ !changes.loading.currentValue &&
178
+ changes.loading.previousValue) {
179
+ if (this.opened === "yes") {
180
+ // 如果提供了固定尺寸,位置已经计算好了,不需要更新
181
+ if (this._width !== null && this._height !== null) {
182
+ this.contentLoad.emit();
183
+ }
184
+ else {
185
+ // 如果正在定位,不需要更新位置
186
+ if (this.positioning === "no") {
187
+ requestAnimationFrame(() => {
188
+ this.updatePopupPosition();
189
+ this.contentLoad.emit();
190
+ });
191
+ }
192
+ else {
193
+ // 如果正在定位,等待定位完成后再触发事件
194
+ const checkPositioning = () => {
195
+ if (this.positioning === "no") {
196
+ this.contentLoad.emit();
197
+ }
198
+ else {
199
+ setTimeout(checkPositioning, 10);
200
+ }
201
+ };
202
+ checkPositioning();
203
+ }
204
+ }
205
+ }
206
+ }
207
+ }
208
+ ngOnDestroy() {
209
+ if (this.popupElement && this.popupElement.nativeElement) {
210
+ this.popupElement.nativeElement.remove();
211
+ }
212
+ window.removeEventListener("click", this.onClickOutside);
213
+ window.removeEventListener("scroll", this.onWindowScroll, true);
214
+ }
215
+ updateTriggerElement() {
216
+ if (this.triggerElementRef) {
217
+ this.currentTriggerElement = this.triggerElementRef;
218
+ }
219
+ else if (this.internalTriggerElement &&
220
+ this.internalTriggerElement.nativeElement) {
221
+ this.currentTriggerElement = this.internalTriggerElement.nativeElement;
222
+ }
223
+ else {
224
+ this.currentTriggerElement = null;
225
+ }
226
+ }
227
+ FIXED_CONTAINER_ID = "rs-smart-popup-fixed-container";
228
+ fixedContainerEl;
229
+ getFixedContainer() {
230
+ let containerElement = document.getElementById(this.FIXED_CONTAINER_ID);
231
+ if (!containerElement) {
232
+ containerElement = document.createElement("div");
233
+ containerElement.setAttribute("id", this.FIXED_CONTAINER_ID);
234
+ document.body.append(containerElement);
235
+ }
236
+ return containerElement;
237
+ }
238
+ opened = "no";
239
+ positioning = "no";
240
+ actualPlacement = "bottom";
241
+ actualPosition = "start";
242
+ open(triggerElement) {
243
+ // 如果传入了触发元素,使用它
244
+ if (triggerElement) {
245
+ this.currentTriggerElement = triggerElement;
246
+ }
247
+ else {
248
+ this.updateTriggerElement();
249
+ }
250
+ if (!this.currentTriggerElement) {
251
+ console.warn("SmartPopupComponent: No trigger element found");
252
+ return;
253
+ }
254
+ // 如果已经打开,且提供了固定尺寸,直接重新计算位置(避免关闭再打开导致的跳动)
255
+ if (this.opened === "yes") {
256
+ if (this._width !== null && this._height !== null) {
257
+ // 有固定尺寸,直接重新计算位置,不关闭弹窗
258
+ // 先隐藏弹窗(positioning = yes)
259
+ this.positioning = "yes";
260
+ this.ref.markForCheck();
261
+ this.ref.detectChanges(); // 立即应用隐藏状态
262
+ requestAnimationFrame(() => {
263
+ // 再次确保触发元素已更新
264
+ this.updateTriggerElement();
265
+ if (this.currentTriggerElement) {
266
+ const triggerRect = this.currentTriggerElement.getBoundingClientRect();
267
+ const popupSize = {
268
+ width: this._width,
269
+ height: this._height,
270
+ };
271
+ const targetInfo = this.placementInfo;
272
+ let finalPlacement = targetInfo.placement;
273
+ const finalPosition = targetInfo.position;
274
+ if (this.autoAdjust) {
275
+ const popupRect = {
276
+ width: popupSize.width,
277
+ height: popupSize.height,
278
+ top: 0,
279
+ left: 0,
280
+ right: popupSize.width,
281
+ bottom: popupSize.height,
282
+ x: 0,
283
+ y: 0,
284
+ toJSON: () => ({}),
285
+ };
286
+ const optimal = this.calculateOptimalPlacement(triggerRect, popupRect, window.innerWidth, window.innerHeight);
287
+ finalPlacement = optimal.placement;
288
+ }
289
+ this.actualPlacement = finalPlacement;
290
+ this.actualPosition = finalPosition;
291
+ const position = this.calculatePositionCoordinates(triggerRect, finalPlacement, finalPosition, popupSize);
292
+ if (this.popupElement && this.popupElement.nativeElement) {
293
+ const popupEl = this.popupElement.nativeElement;
294
+ popupEl.style.top = position.top;
295
+ popupEl.style.left = position.left;
296
+ popupEl.style.transform = `translate(${position.translateX}, ${position.translateY})`;
297
+ popupEl.offsetHeight; // 强制同步应用样式
298
+ // 位置设置完成后,再显示弹窗
299
+ requestAnimationFrame(() => {
300
+ this.positioning = "no";
301
+ this.ref.markForCheck();
302
+ });
303
+ }
304
+ }
305
+ });
306
+ return;
307
+ }
308
+ else {
309
+ // 没有固定尺寸,先关闭再打开
310
+ this.opened = "no";
311
+ this.positioning = "no";
312
+ this.ref.markForCheck();
313
+ requestAnimationFrame(() => {
314
+ this.updateTriggerElement();
315
+ this.openPopup();
316
+ });
317
+ return;
318
+ }
319
+ }
320
+ // 确保触发元素已更新
321
+ this.updateTriggerElement();
322
+ this.openPopup();
323
+ }
324
+ openPopup() {
325
+ // 再次确保触发元素已更新
326
+ this.updateTriggerElement();
327
+ if (!this.currentTriggerElement) {
328
+ console.warn("SmartPopupComponent: No trigger element found when opening popup");
329
+ return;
330
+ }
331
+ // 如果提供了固定尺寸,可以在打开前就计算好位置
332
+ if (this._width !== null && this._height !== null) {
333
+ // 先计算位置(不依赖DOM渲染,直接根据triggerElement计算)
334
+ const triggerRect = this.currentTriggerElement.getBoundingClientRect();
335
+ const viewportWidth = window.innerWidth;
336
+ const viewportHeight = window.innerHeight;
337
+ const popupSize = {
338
+ width: this._width,
339
+ height: this._height,
340
+ };
341
+ const popupRect = {
342
+ width: popupSize.width,
343
+ height: popupSize.height,
344
+ top: 0,
345
+ left: 0,
346
+ right: popupSize.width,
347
+ bottom: popupSize.height,
348
+ x: 0,
349
+ y: 0,
350
+ toJSON: () => ({}),
351
+ };
352
+ const targetInfo = this.placementInfo;
353
+ let finalPlacement = targetInfo.placement;
354
+ // 始终使用用户指定的 position,autoAdjust 只调整 placement(方向),不改变对齐方式
355
+ const finalPosition = targetInfo.position;
356
+ if (this.autoAdjust) {
357
+ const optimal = this.calculateOptimalPlacement(triggerRect, popupRect, viewportWidth, viewportHeight);
358
+ // autoAdjust 只调整 placement(上下左右),保持用户指定的 position(对齐方式)
359
+ finalPlacement = optimal.placement;
360
+ // finalPosition 保持用户指定的值,不改变
361
+ }
362
+ this.actualPlacement = finalPlacement;
363
+ this.actualPosition = finalPosition;
364
+ // 计算位置坐标(不依赖DOM,直接计算)
365
+ const position = this.calculatePositionCoordinates(triggerRect, finalPlacement, finalPosition, popupSize);
366
+ // 先设置打开状态,但保持隐藏(positioning = yes)
367
+ // 注意:先设置 positioning = yes,再设置 opened = yes,确保弹窗在位置计算完成前始终隐藏
368
+ this.positioning = "yes";
369
+ this.opened = "yes";
370
+ this.openChange.emit(true);
371
+ this.ref.markForCheck();
372
+ // 强制更新视图,确保DOM已更新,但弹窗仍然隐藏(因为 positioning = yes)
373
+ this.ref.detectChanges();
374
+ // 设置位置的函数
375
+ const setPosition = () => {
376
+ if (this.popupElement && this.popupElement.nativeElement) {
377
+ const popupEl = this.popupElement.nativeElement;
378
+ // 立即设置尺寸和位置
379
+ popupEl.style.width = this._width + "px";
380
+ popupEl.style.height = this._height + "px";
381
+ popupEl.style.minWidth = this._width + "px";
382
+ popupEl.style.maxWidth = this._width + "px";
383
+ popupEl.style.minHeight = this._height + "px";
384
+ popupEl.style.maxHeight = this._height + "px";
385
+ popupEl.style.top = position.top;
386
+ popupEl.style.left = position.left;
387
+ popupEl.style.transform = `translate(${position.translateX}, ${position.translateY})`;
388
+ // 强制同步应用样式,确保位置已设置
389
+ popupEl.offsetHeight;
390
+ // 位置设置完成后,再显示弹窗(设置 positioning = no)
391
+ // 使用 requestAnimationFrame 确保样式已应用
392
+ requestAnimationFrame(() => {
393
+ this.positioning = "no";
394
+ this.ref.markForCheck();
395
+ });
396
+ }
397
+ };
398
+ // 如果元素已经存在,立即设置位置
399
+ if (this.popupElement && this.popupElement.nativeElement) {
400
+ setPosition();
401
+ }
402
+ else {
403
+ // 如果元素还不存在,等待渲染后再设置位置
404
+ requestAnimationFrame(() => {
405
+ if (this.popupElement && this.popupElement.nativeElement) {
406
+ setPosition();
407
+ }
408
+ });
409
+ }
410
+ }
411
+ else {
412
+ // 没有固定尺寸,需要等待DOM渲染
413
+ this.positioning = "yes";
414
+ this.ref.markForCheck();
415
+ // 立即强制更新视图,确保DOM已更新
416
+ this.ref.detectChanges();
417
+ // 使用 requestAnimationFrame 确保在浏览器渲染之前更新位置
418
+ requestAnimationFrame(() => {
419
+ // 确保元素已渲染并获取尺寸
420
+ if (this.popupElement && this.popupElement.nativeElement) {
421
+ // 强制重新计算布局,确保元素尺寸已计算
422
+ this.popupElement.nativeElement.offsetHeight;
423
+ // 计算并设置位置
424
+ this.updatePopupPosition();
425
+ // 位置设置完成后,移除定位状态,显示弹窗
426
+ this.positioning = "no";
427
+ this.ref.markForCheck();
428
+ }
429
+ });
430
+ }
431
+ window.addEventListener("click", this.onClickOutside);
432
+ window.addEventListener("scroll", this.onWindowScroll, true);
433
+ }
434
+ close() {
435
+ if (this.opened === "no") {
436
+ return;
437
+ }
438
+ this.opened = "no";
439
+ this.positioning = "no";
440
+ this.openChange.emit(false);
441
+ window.removeEventListener("click", this.onClickOutside);
442
+ window.removeEventListener("scroll", this.onWindowScroll, true);
443
+ this.ref.markForCheck();
444
+ }
445
+ toggle() {
446
+ if (this.opened === "yes") {
447
+ this.close();
448
+ }
449
+ else {
450
+ this.open();
451
+ }
452
+ }
453
+ onClickTrigger(event) {
454
+ event.stopPropagation();
455
+ this.toggle();
456
+ }
457
+ onClickOutside = (event) => {
458
+ const target = event.target;
459
+ const isClickOnTrigger = this.currentTriggerElement && this.currentTriggerElement.contains(target);
460
+ const isClickOnPopup = this.popupElement &&
461
+ this.popupElement.nativeElement &&
462
+ this.popupElement.nativeElement.contains(target);
463
+ if (!isClickOnTrigger && !isClickOnPopup) {
464
+ this.close();
465
+ }
466
+ };
467
+ onWindowScroll = (event) => {
468
+ if (this.opened === "yes") {
469
+ // 检查滚动是否发生在弹窗内部
470
+ const target = event.target;
471
+ const isScrollOnPopup = this.popupElement &&
472
+ this.popupElement.nativeElement &&
473
+ this.popupElement.nativeElement.contains(target);
474
+ // 如果滚动发生在弹窗内部,不更新位置(避免不必要的计算)
475
+ // 如果滚动发生在外部,更新位置(因为触发元素位置可能改变)
476
+ if (!isScrollOnPopup) {
477
+ this.updatePopupPosition();
478
+ }
479
+ }
480
+ };
481
+ // 处理弹窗内部的滚轮事件
482
+ onPopupWheel(event) {
483
+ // 阻止滚轮事件冒泡,避免触发外部滚动处理
484
+ event.stopPropagation();
485
+ }
486
+ // 处理弹窗内部的滚动事件
487
+ onPopupScroll(event) {
488
+ // 阻止滚动事件冒泡,避免触发外部滚动处理
489
+ event.stopPropagation();
490
+ }
491
+ // 计算并设置位置(支持固定尺寸,可在打开前调用)
492
+ calculateAndSetPosition() {
493
+ if (!this.currentTriggerElement) {
494
+ return;
495
+ }
496
+ const triggerRect = this.currentTriggerElement.getBoundingClientRect();
497
+ const viewportWidth = window.innerWidth;
498
+ const viewportHeight = window.innerHeight;
499
+ // 获取弹窗尺寸(优先使用固定尺寸)
500
+ const popupSize = {
501
+ width: this._width !== null
502
+ ? this._width
503
+ : this.popupElement && this.popupElement.nativeElement
504
+ ? this.popupElement.nativeElement.offsetWidth
505
+ : 300,
506
+ height: this._height !== null
507
+ ? this._height
508
+ : this.popupElement && this.popupElement.nativeElement
509
+ ? this.popupElement.nativeElement.offsetHeight
510
+ : 200,
511
+ };
512
+ // 创建虚拟的 popupRect(用于计算)
513
+ const popupRect = {
514
+ width: popupSize.width,
515
+ height: popupSize.height,
516
+ top: 0,
517
+ left: 0,
518
+ right: popupSize.width,
519
+ bottom: popupSize.height,
520
+ x: 0,
521
+ y: 0,
522
+ toJSON: () => ({}),
523
+ };
524
+ // 获取目标位置信息
525
+ const targetInfo = this.placementInfo;
526
+ let finalPlacement = targetInfo.placement;
527
+ // 始终使用用户指定的 position,autoAdjust 只调整 placement(方向),不改变对齐方式
528
+ const finalPosition = targetInfo.position;
529
+ if (this.autoAdjust) {
530
+ const optimal = this.calculateOptimalPlacement(triggerRect, popupRect, viewportWidth, viewportHeight);
531
+ // autoAdjust 只调整 placement(上下左右),保持用户指定的 position(对齐方式)
532
+ finalPlacement = optimal.placement;
533
+ // finalPosition 保持用户指定的值,不改变
534
+ }
535
+ this.actualPlacement = finalPlacement;
536
+ this.actualPosition = finalPosition;
537
+ // 立即设置位置(即使元素还没完全渲染,也要先设置位置)
538
+ if (this.popupElement && this.popupElement.nativeElement) {
539
+ this.setPopupPosition(triggerRect, popupRect, finalPlacement, finalPosition, popupSize);
540
+ // 强制应用样式,确保位置立即生效
541
+ this.ref.detectChanges();
542
+ }
543
+ else {
544
+ // 如果元素还不存在,延迟设置位置
545
+ setTimeout(() => {
546
+ if (this.popupElement && this.popupElement.nativeElement) {
547
+ this.setPopupPosition(triggerRect, popupRect, finalPlacement, finalPosition, popupSize);
548
+ this.ref.detectChanges();
549
+ }
550
+ }, 0);
551
+ }
552
+ this.ref.markForCheck();
553
+ }
554
+ updatePopupPosition() {
555
+ if (!this.currentTriggerElement ||
556
+ !this.popupElement ||
557
+ !this.popupElement.nativeElement) {
558
+ return;
559
+ }
560
+ // 如果提供了固定尺寸,使用固定尺寸计算
561
+ if (this._width !== null && this._height !== null) {
562
+ this.calculateAndSetPosition();
563
+ return;
564
+ }
565
+ const triggerRect = this.currentTriggerElement.getBoundingClientRect();
566
+ const popupRect = this.popupElement.nativeElement.getBoundingClientRect();
567
+ const viewportWidth = window.innerWidth;
568
+ const viewportHeight = window.innerHeight;
569
+ // 获取目标位置信息
570
+ const targetInfo = this.placementInfo;
571
+ let finalPlacement = targetInfo.placement;
572
+ // 始终使用用户指定的 position,autoAdjust 只调整 placement(方向),不改变对齐方式
573
+ const finalPosition = targetInfo.position;
574
+ if (this.autoAdjust) {
575
+ const optimal = this.calculateOptimalPlacement(triggerRect, popupRect, viewportWidth, viewportHeight);
576
+ // autoAdjust 只调整 placement(上下左右),保持用户指定的 position(对齐方式)
577
+ finalPlacement = optimal.placement;
578
+ // finalPosition 保持用户指定的值,不改变
579
+ }
580
+ this.actualPlacement = finalPlacement;
581
+ this.actualPosition = finalPosition;
582
+ this.setPopupPosition(triggerRect, popupRect, finalPlacement, finalPosition);
583
+ this.ref.markForCheck();
584
+ }
585
+ calculateOptimalPlacement(triggerRect, popupRect, viewportWidth, viewportHeight) {
586
+ const targetInfo = this.placementInfo;
587
+ const preferred = targetInfo.placement;
588
+ const preferredPosition = targetInfo.position;
589
+ const space = {
590
+ top: triggerRect.top,
591
+ bottom: viewportHeight - triggerRect.bottom,
592
+ left: triggerRect.left,
593
+ right: viewportWidth - triggerRect.right,
594
+ };
595
+ const popupSize = {
596
+ width: popupRect.width || 300,
597
+ height: popupRect.height || 200, // 默认高度
598
+ };
599
+ // 检查首选位置是否有足够空间
600
+ const hasEnoughSpace = (placement) => {
601
+ switch (placement) {
602
+ case "top":
603
+ return space.top >= popupSize.height + this.offset;
604
+ case "bottom":
605
+ return space.bottom >= popupSize.height + this.offset;
606
+ case "left":
607
+ return space.left >= popupSize.width + this.offset;
608
+ case "right":
609
+ return space.right >= popupSize.width + this.offset;
610
+ }
611
+ };
612
+ // 检查指定位置是否有足够空间
613
+ const checkPositionSpace = (placement, position) => {
614
+ if (placement === "top" || placement === "bottom") {
615
+ // 对于上下位置,检查水平对齐空间
616
+ const popupWidth = popupSize.width;
617
+ const spaceLeft = triggerRect.left;
618
+ const spaceRight = viewportWidth - triggerRect.right;
619
+ if (position === "center") {
620
+ return spaceLeft >= popupWidth / 2 && spaceRight >= popupWidth / 2;
621
+ }
622
+ else if (position === "start") {
623
+ return spaceLeft >= popupWidth;
624
+ }
625
+ else {
626
+ // end
627
+ return spaceRight >= popupWidth;
628
+ }
629
+ }
630
+ else {
631
+ // 对于左右位置,检查垂直对齐空间
632
+ const popupHeight = popupSize.height;
633
+ const spaceTop = triggerRect.top;
634
+ const spaceBottom = viewportHeight - triggerRect.bottom;
635
+ if (position === "center") {
636
+ return spaceTop >= popupHeight / 2 && spaceBottom >= popupHeight / 2;
637
+ }
638
+ else if (position === "start") {
639
+ return spaceTop >= popupHeight;
640
+ }
641
+ else {
642
+ // end
643
+ return spaceBottom >= popupHeight;
644
+ }
645
+ }
646
+ };
647
+ // 计算位置是否合适(考虑水平/垂直对齐)
648
+ const calculatePosition = (placement) => {
649
+ if (placement === "top" || placement === "bottom") {
650
+ // 对于上下位置,检查水平对齐
651
+ const popupWidth = popupSize.width;
652
+ const spaceLeft = triggerRect.left;
653
+ const spaceRight = viewportWidth - triggerRect.right;
654
+ // 优先尝试使用用户指定的position
655
+ if (checkPositionSpace(placement, preferredPosition)) {
656
+ return preferredPosition;
657
+ }
658
+ // 如果中心对齐有足够空间,优先使用中心
659
+ if (spaceLeft >= popupWidth / 2 && spaceRight >= popupWidth / 2) {
660
+ return "center";
661
+ }
662
+ // 如果左侧空间不足,使用右对齐
663
+ if (spaceLeft < popupWidth / 2) {
664
+ return "end";
665
+ }
666
+ // 如果右侧空间不足,使用左对齐
667
+ if (spaceRight < popupWidth / 2) {
668
+ return "start";
669
+ }
670
+ return preferredPosition;
671
+ }
672
+ else {
673
+ // 对于左右位置,检查垂直对齐
674
+ const popupHeight = popupSize.height;
675
+ const spaceTop = triggerRect.top;
676
+ const spaceBottom = viewportHeight - triggerRect.bottom;
677
+ // 优先尝试使用用户指定的position
678
+ if (checkPositionSpace(placement, preferredPosition)) {
679
+ return preferredPosition;
680
+ }
681
+ // 如果中心对齐有足够空间,优先使用中心
682
+ if (spaceTop >= popupHeight / 2 && spaceBottom >= popupHeight / 2) {
683
+ return "center";
684
+ }
685
+ // 如果上方空间不足,使用下对齐
686
+ if (spaceTop < popupHeight / 2) {
687
+ return "end";
688
+ }
689
+ // 如果下方空间不足,使用上对齐
690
+ if (spaceBottom < popupHeight / 2) {
691
+ return "start";
692
+ }
693
+ return preferredPosition;
694
+ }
695
+ };
696
+ // 如果首选位置有足够空间,优先使用用户指定的position
697
+ if (hasEnoughSpace(preferred)) {
698
+ // 即使指定的position空间不足,也尽量保持用户指定的position
699
+ // 边界约束逻辑会在 calculatePositionCoordinates 中处理位置调整
700
+ // 这样可以保持用户期望的对齐方式
701
+ return {
702
+ placement: preferred,
703
+ position: preferredPosition,
704
+ };
705
+ }
706
+ // 根据首选位置选择备选位置
707
+ const alternatives = {
708
+ bottom: ["top", "right", "left"],
709
+ top: ["bottom", "right", "left"],
710
+ right: ["left", "bottom", "top"],
711
+ left: ["right", "bottom", "top"],
712
+ };
713
+ const altList = alternatives[preferred] || [];
714
+ // 尝试找到第一个有足够空间的备选位置
715
+ for (const alt of altList) {
716
+ if (hasEnoughSpace(alt)) {
717
+ return {
718
+ placement: alt,
719
+ position: calculatePosition(alt),
720
+ };
721
+ }
722
+ }
723
+ // 如果都没有足够空间,选择空间最大的位置
724
+ const maxSpace = Math.max(space.top, space.bottom, space.left, space.right);
725
+ let bestPlacement;
726
+ if (maxSpace === space.top)
727
+ bestPlacement = "top";
728
+ else if (maxSpace === space.bottom)
729
+ bestPlacement = "bottom";
730
+ else if (maxSpace === space.left)
731
+ bestPlacement = "left";
732
+ else
733
+ bestPlacement = "right";
734
+ return {
735
+ placement: bestPlacement,
736
+ position: calculatePosition(bestPlacement),
737
+ };
738
+ }
739
+ // 计算位置坐标(纯计算,不依赖DOM,包含边界约束)
740
+ // 注意:弹窗使用 position: fixed,所以坐标应该是相对于视口的,不需要加上滚动偏移
741
+ calculatePositionCoordinates(triggerRect, placement, position, popupSize) {
742
+ const triggerWidth = triggerRect.width;
743
+ const triggerHeight = triggerRect.height;
744
+ const viewportWidth = window.innerWidth;
745
+ const viewportHeight = window.innerHeight;
746
+ let top = 0;
747
+ let left = 0;
748
+ let translateX = "0";
749
+ let translateY = "0";
750
+ switch (placement) {
751
+ case "top":
752
+ // 弹窗在触发元素上方
753
+ top = triggerRect.top - this.offset;
754
+ translateY = "-100%";
755
+ if (position === "start") {
756
+ // 左对齐
757
+ left = triggerRect.left;
758
+ translateX = "0";
759
+ }
760
+ else if (position === "center") {
761
+ // 居中对齐
762
+ left = triggerRect.left + triggerWidth / 2;
763
+ translateX = "-50%";
764
+ }
765
+ else {
766
+ // 右对齐(end)
767
+ left = triggerRect.right;
768
+ translateX = "-100%";
769
+ }
770
+ // 边界约束:确保弹窗不超出视口
771
+ const popupTop = top - popupSize.height;
772
+ if (popupTop < 0) {
773
+ top = popupSize.height + 8; // 距离顶部至少8px
774
+ translateY = "0"; // 取消向上偏移
775
+ }
776
+ // 水平边界约束:尽量保持原始对齐方式
777
+ if (position === "start") {
778
+ // 左对齐:如果超出右边界,调整到右边界;如果超出左边界,调整到左边界
779
+ if (left + popupSize.width > viewportWidth) {
780
+ left = viewportWidth - popupSize.width - 8;
781
+ }
782
+ if (left < 8) {
783
+ left = 8;
784
+ }
785
+ }
786
+ else if (position === "center") {
787
+ // 居中对齐:计算实际左边界
788
+ const actualLeft = left - popupSize.width / 2;
789
+ if (actualLeft < 8) {
790
+ // 左边界不足,尽量保持居中,但调整到不超出左边界
791
+ left = 8 + popupSize.width / 2;
792
+ translateX = "-50%";
793
+ }
794
+ else if (actualLeft + popupSize.width > viewportWidth - 8) {
795
+ // 右边界不足,尽量保持居中,但调整到不超出右边界
796
+ left = viewportWidth - 8 - popupSize.width / 2;
797
+ translateX = "-50%";
798
+ }
799
+ }
800
+ else {
801
+ // 右对齐(end):弹窗右边缘与触发元素右边缘对齐
802
+ // left 是弹窗的右边缘位置(因为 translateX = "-100%")
803
+ // 弹窗的左边缘在 left - popupSize.width
804
+ const popupLeftEdge = left - popupSize.width;
805
+ // 如果弹窗左边缘超出左边界,调整位置但保持右对齐
806
+ if (popupLeftEdge < 8) {
807
+ left = 8 + popupSize.width;
808
+ translateX = "-100%";
809
+ }
810
+ // 如果弹窗右边缘超出右边界,调整位置但保持右对齐
811
+ if (left > viewportWidth - 8) {
812
+ left = viewportWidth - 8;
813
+ translateX = "-100%";
814
+ }
815
+ }
816
+ break;
817
+ case "bottom":
818
+ // 弹窗在触发元素下方
819
+ top = triggerRect.bottom + this.offset;
820
+ if (position === "start") {
821
+ // 左对齐
822
+ left = triggerRect.left;
823
+ translateX = "0";
824
+ }
825
+ else if (position === "center") {
826
+ // 居中对齐
827
+ left = triggerRect.left + triggerWidth / 2;
828
+ translateX = "-50%";
829
+ }
830
+ else {
831
+ // 右对齐(end)
832
+ left = triggerRect.right;
833
+ translateX = "-100%";
834
+ }
835
+ // 边界约束:确保弹窗不超出视口
836
+ if (top + popupSize.height > viewportHeight) {
837
+ top = viewportHeight - popupSize.height - 8; // 距离底部至少8px
838
+ }
839
+ // 水平边界约束:尽量保持原始对齐方式
840
+ if (position === "start") {
841
+ // 左对齐:如果超出右边界,调整到右边界;如果超出左边界,调整到左边界
842
+ if (left + popupSize.width > viewportWidth) {
843
+ left = viewportWidth - popupSize.width - 8;
844
+ }
845
+ if (left < 8) {
846
+ left = 8;
847
+ }
848
+ }
849
+ else if (position === "center") {
850
+ // 居中对齐:计算实际左边界
851
+ const actualLeft = left - popupSize.width / 2;
852
+ if (actualLeft < 8) {
853
+ // 左边界不足,尽量保持居中,但调整到不超出左边界
854
+ left = 8 + popupSize.width / 2;
855
+ translateX = "-50%";
856
+ }
857
+ else if (actualLeft + popupSize.width > viewportWidth - 8) {
858
+ // 右边界不足,尽量保持居中,但调整到不超出右边界
859
+ left = viewportWidth - 8 - popupSize.width / 2;
860
+ translateX = "-50%";
861
+ }
862
+ }
863
+ else {
864
+ // 右对齐(end):弹窗右边缘与触发元素右边缘对齐
865
+ // left 是弹窗的右边缘位置(因为 translateX = "-100%")
866
+ // 弹窗的左边缘在 left - popupSize.width
867
+ const popupLeftEdge = left - popupSize.width;
868
+ // 优先保持与触发元素右对齐,只在必要时调整
869
+ // 如果弹窗右边缘超出右边界,调整位置但保持右对齐
870
+ if (left > viewportWidth - 8) {
871
+ left = viewportWidth - 8;
872
+ translateX = "-100%";
873
+ }
874
+ // 如果弹窗左边缘超出左边界,且弹窗可以完全显示在视口内,才调整位置
875
+ // 这样可以确保在可能的情况下,弹窗始终与触发元素右对齐
876
+ if (popupLeftEdge < 8 && popupSize.width < viewportWidth - 16) {
877
+ // 只有当弹窗可以完全显示在视口内时,才调整位置
878
+ // 否则保持与触发元素右对齐,即使部分超出左边界
879
+ left = 8 + popupSize.width;
880
+ translateX = "-100%";
881
+ }
882
+ }
883
+ break;
884
+ case "left":
885
+ // 弹窗在触发元素左侧
886
+ left = triggerRect.left - this.offset;
887
+ translateX = "-100%";
888
+ if (position === "start") {
889
+ // 上对齐
890
+ top = triggerRect.top;
891
+ translateY = "0";
892
+ }
893
+ else if (position === "center") {
894
+ // 居中对齐
895
+ top = triggerRect.top + triggerHeight / 2;
896
+ translateY = "-50%";
897
+ }
898
+ else {
899
+ // 下对齐(end)
900
+ top = triggerRect.bottom;
901
+ translateY = "-100%";
902
+ }
903
+ // 边界约束:确保弹窗不超出视口
904
+ const popupLeftSide = left - popupSize.width;
905
+ if (popupLeftSide < 0) {
906
+ left = 8; // 距离左侧至少8px
907
+ translateX = "0"; // 取消向左偏移
908
+ }
909
+ // 垂直边界约束:尽量保持原始对齐方式
910
+ if (position === "start") {
911
+ // 上对齐:如果超出下边界,调整到下边界;如果超出上边界,调整到上边界
912
+ if (top + popupSize.height > viewportHeight) {
913
+ top = viewportHeight - popupSize.height - 8;
914
+ }
915
+ if (top < 8) {
916
+ top = 8;
917
+ }
918
+ }
919
+ else if (position === "center") {
920
+ // 居中对齐:计算实际上边界
921
+ const actualTop = top - popupSize.height / 2;
922
+ if (actualTop < 8) {
923
+ // 上边界不足,尽量保持居中,但调整到不超出上边界
924
+ top = 8 + popupSize.height / 2;
925
+ translateY = "-50%";
926
+ }
927
+ else if (actualTop + popupSize.height > viewportHeight - 8) {
928
+ // 下边界不足,尽量保持居中,但调整到不超出下边界
929
+ top = viewportHeight - 8 - popupSize.height / 2;
930
+ translateY = "-50%";
931
+ }
932
+ }
933
+ else {
934
+ // 下对齐(end):如果超出上边界,调整到上边界;如果超出下边界,调整到下边界
935
+ const actualTop = top - popupSize.height;
936
+ if (actualTop < 8) {
937
+ top = 8 + popupSize.height;
938
+ translateY = "-100%";
939
+ }
940
+ if (top > viewportHeight - 8) {
941
+ top = viewportHeight - 8;
942
+ translateY = "-100%";
943
+ }
944
+ }
945
+ break;
946
+ case "right":
947
+ // 弹窗在触发元素右侧
948
+ left = triggerRect.right + this.offset;
949
+ translateX = "0";
950
+ if (position === "start") {
951
+ // 上对齐
952
+ top = triggerRect.top;
953
+ translateY = "0";
954
+ }
955
+ else if (position === "center") {
956
+ // 居中对齐
957
+ top = triggerRect.top + triggerHeight / 2;
958
+ translateY = "-50%";
959
+ }
960
+ else {
961
+ // 下对齐(end)
962
+ top = triggerRect.bottom;
963
+ translateY = "-100%";
964
+ }
965
+ // 边界约束:确保弹窗不超出视口
966
+ if (left + popupSize.width > viewportWidth) {
967
+ left = viewportWidth - popupSize.width - 8; // 距离右侧至少8px
968
+ }
969
+ // 垂直边界约束:尽量保持原始对齐方式
970
+ if (position === "start") {
971
+ // 上对齐:如果超出下边界,调整到下边界;如果超出上边界,调整到上边界
972
+ if (top + popupSize.height > viewportHeight) {
973
+ top = viewportHeight - popupSize.height - 8;
974
+ }
975
+ if (top < 8) {
976
+ top = 8;
977
+ }
978
+ }
979
+ else if (position === "center") {
980
+ // 居中对齐:计算实际上边界
981
+ const actualTop = top - popupSize.height / 2;
982
+ if (actualTop < 8) {
983
+ // 上边界不足,尽量保持居中,但调整到不超出上边界
984
+ top = 8 + popupSize.height / 2;
985
+ translateY = "-50%";
986
+ }
987
+ else if (actualTop + popupSize.height > viewportHeight - 8) {
988
+ // 下边界不足,尽量保持居中,但调整到不超出下边界
989
+ top = viewportHeight - 8 - popupSize.height / 2;
990
+ translateY = "-50%";
991
+ }
992
+ }
993
+ else {
994
+ // 下对齐(end):如果超出上边界,调整到上边界;如果超出下边界,调整到下边界
995
+ const actualTop = top - popupSize.height;
996
+ if (actualTop < 8) {
997
+ top = 8 + popupSize.height;
998
+ translateY = "-100%";
999
+ }
1000
+ if (top > viewportHeight - 8) {
1001
+ top = viewportHeight - 8;
1002
+ translateY = "-100%";
1003
+ }
1004
+ }
1005
+ break;
1006
+ }
1007
+ return {
1008
+ top: top + "px",
1009
+ left: left + "px",
1010
+ translateX,
1011
+ translateY,
1012
+ };
1013
+ }
1014
+ setPopupPosition(triggerRect, popupRect, placement, position, popupSize) {
1015
+ const popupEl = this.popupElement && this.popupElement.nativeElement
1016
+ ? this.popupElement.nativeElement
1017
+ : null;
1018
+ if (!popupEl) {
1019
+ return; // 如果元素还不存在,无法设置位置
1020
+ }
1021
+ // 优先使用传入的固定尺寸,否则从DOM获取
1022
+ const popupWidth = popupSize && popupSize.width !== null && popupSize.width !== undefined
1023
+ ? popupSize.width
1024
+ : popupRect.width || popupEl.offsetWidth || 300;
1025
+ const popupHeight = popupSize && popupSize.height !== null && popupSize.height !== undefined
1026
+ ? popupSize.height
1027
+ : popupRect.height || popupEl.offsetHeight || 200;
1028
+ const coordinates = this.calculatePositionCoordinates(triggerRect, placement, position, { width: popupWidth, height: popupHeight });
1029
+ popupEl.style.top = coordinates.top;
1030
+ popupEl.style.left = coordinates.left;
1031
+ popupEl.style.transform = `translate(${coordinates.translateX}, ${coordinates.translateY})`;
1032
+ }
1033
+ onWindowResize() {
1034
+ if (this.opened === "yes") {
1035
+ this.updatePopupPosition();
1036
+ }
1037
+ }
1038
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: SmartPopupComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
1039
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: SmartPopupComponent, selector: "rs-smart-popup", inputs: { _placement: ["placement", "_placement"], _offset: ["offset", "_offset"], _autoAdjust: ["autoAdjust", "_autoAdjust"], triggerElementRef: ["triggerElement", "triggerElementRef"], loading: "loading", _open: ["open", "_open"], _width: ["width", "_width"], _height: ["height", "_height"] }, outputs: { openChange: "openChange", contentLoad: "contentLoad" }, host: { listeners: { "window:resize": "onWindowResize()" } }, viewQueries: [{ propertyName: "internalTriggerElement", first: true, predicate: ["trigger"], descendants: true }, { propertyName: "popupElement", first: true, predicate: ["popup"], descendants: true }], usesOnChanges: true, ngImport: i0, 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 (wheel)=\"onPopupWheel($event)\"\r\n (scroll)=\"onPopupScroll($event)\"\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", 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 #00000040;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}::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%;animation:spin 1s linear infinite;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}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content .rs-smart-popup-body{width:100%}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
1040
+ }
1041
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: SmartPopupComponent, decorators: [{
1042
+ type: Component,
1043
+ args: [{ selector: "rs-smart-popup", 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 (wheel)=\"onPopupWheel($event)\"\r\n (scroll)=\"onPopupScroll($event)\"\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", 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 #00000040;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}::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%;animation:spin 1s linear infinite;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}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}::ng-deep #rs-smart-popup-fixed-container .rs-smart-popup-content .rs-smart-popup-body{width:100%}\n"] }]
1044
+ }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { _placement: [{
1045
+ type: Input,
1046
+ args: ["placement"]
1047
+ }], _offset: [{
1048
+ type: Input,
1049
+ args: ["offset"]
1050
+ }], _autoAdjust: [{
1051
+ type: Input,
1052
+ args: ["autoAdjust"]
1053
+ }], triggerElementRef: [{
1054
+ type: Input,
1055
+ args: ["triggerElement"]
1056
+ }], loading: [{
1057
+ type: Input,
1058
+ args: ["loading"]
1059
+ }], _open: [{
1060
+ type: Input,
1061
+ args: ["open"]
1062
+ }], _width: [{
1063
+ type: Input,
1064
+ args: ["width"]
1065
+ }], _height: [{
1066
+ type: Input,
1067
+ args: ["height"]
1068
+ }], openChange: [{
1069
+ type: Output,
1070
+ args: ["openChange"]
1071
+ }], contentLoad: [{
1072
+ type: Output,
1073
+ args: ["contentLoad"]
1074
+ }], internalTriggerElement: [{
1075
+ type: ViewChild,
1076
+ args: ["trigger", { static: false }]
1077
+ }], popupElement: [{
1078
+ type: ViewChild,
1079
+ args: ["popup", { static: false }]
1080
+ }], onWindowResize: [{
1081
+ type: HostListener,
1082
+ args: ["window:resize"]
1083
+ }] } });
1084
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.component.js","sourceRoot":"","sources":["../../../../../projects/raise-common-lib/src/lib/smart-popup/index.component.ts","../../../../../projects/raise-common-lib/src/lib/smart-popup/index.component.html"],"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        if (this._width !== null && this._height !== null) {\r\n          // 先隐藏弹窗（positioning = yes）\r\n          this.positioning = \"yes\";\r\n          this.ref.markForCheck();\r\n          this.ref.detectChanges(); // 立即应用隐藏状态\r\n          \r\n          requestAnimationFrame(() => {\r\n            if (this.currentTriggerElement) {\r\n              const triggerRect = this.currentTriggerElement.getBoundingClientRect();\r\n              const popupSize = {\r\n                width: this._width!,\r\n                height: this._height!,\r\n              };\r\n              const targetInfo = this.placementInfo;\r\n              let finalPlacement = targetInfo.placement;\r\n              const finalPosition = targetInfo.position;\r\n\r\n              if (this.autoAdjust) {\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                const optimal = this.calculateOptimalPlacement(\r\n                  triggerRect,\r\n                  popupRect,\r\n                  window.innerWidth,\r\n                  window.innerHeight\r\n                );\r\n                finalPlacement = optimal.placement;\r\n              }\r\n\r\n              this.actualPlacement = finalPlacement;\r\n              this.actualPosition = finalPosition;\r\n\r\n              const position = this.calculatePositionCoordinates(\r\n                triggerRect,\r\n                finalPlacement,\r\n                finalPosition,\r\n                popupSize\r\n              );\r\n\r\n              if (this.popupElement && this.popupElement.nativeElement) {\r\n                const popupEl = this.popupElement.nativeElement;\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                popupEl.offsetHeight; // 强制同步应用样式\r\n                \r\n                // 位置设置完成后，再显示弹窗\r\n                requestAnimationFrame(() => {\r\n                  this.positioning = \"no\";\r\n                  this.ref.markForCheck();\r\n                });\r\n              }\r\n            }\r\n          });\r\n        } else {\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\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      if (this._width !== null && this._height !== null) {\r\n        // 有固定尺寸，直接重新计算位置，不关闭弹窗\r\n        // 先隐藏弹窗（positioning = yes）\r\n        this.positioning = \"yes\";\r\n        this.ref.markForCheck();\r\n        this.ref.detectChanges(); // 立即应用隐藏状态\r\n        \r\n        requestAnimationFrame(() => {\r\n          // 再次确保触发元素已更新\r\n          this.updateTriggerElement();\r\n          if (this.currentTriggerElement) {\r\n            const triggerRect = this.currentTriggerElement.getBoundingClientRect();\r\n            const popupSize = {\r\n              width: this._width!,\r\n              height: this._height!,\r\n            };\r\n            const targetInfo = this.placementInfo;\r\n            let finalPlacement = targetInfo.placement;\r\n            const finalPosition = targetInfo.position;\r\n\r\n            if (this.autoAdjust) {\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              const optimal = this.calculateOptimalPlacement(\r\n                triggerRect,\r\n                popupRect,\r\n                window.innerWidth,\r\n                window.innerHeight\r\n              );\r\n              finalPlacement = optimal.placement;\r\n            }\r\n\r\n            this.actualPlacement = finalPlacement;\r\n            this.actualPosition = finalPosition;\r\n\r\n            const position = this.calculatePositionCoordinates(\r\n              triggerRect,\r\n              finalPlacement,\r\n              finalPosition,\r\n              popupSize\r\n            );\r\n\r\n            if (this.popupElement && this.popupElement.nativeElement) {\r\n              const popupEl = this.popupElement.nativeElement;\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              popupEl.offsetHeight; // 强制同步应用样式\r\n              \r\n              // 位置设置完成后，再显示弹窗\r\n              requestAnimationFrame(() => {\r\n                this.positioning = \"no\";\r\n                this.ref.markForCheck();\r\n              });\r\n            }\r\n          }\r\n        });\r\n        return;\r\n      } else {\r\n        // 没有固定尺寸，先关闭再打开\r\n        this.opened = \"no\";\r\n        this.positioning = \"no\";\r\n        this.ref.markForCheck();\r\n        requestAnimationFrame(() => {\r\n          this.updateTriggerElement();\r\n          this.openPopup();\r\n        });\r\n        return;\r\n      }\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      // 始终使用用户指定的 position，autoAdjust 只调整 placement（方向），不改变对齐方式\r\n      const 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        // autoAdjust 只调整 placement（上下左右），保持用户指定的 position（对齐方式）\r\n        finalPlacement = optimal.placement;\r\n        // finalPosition 保持用户指定的值，不改变\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      // 注意：先设置 positioning = yes，再设置 opened = yes，确保弹窗在位置计算完成前始终隐藏\r\n      this.positioning = \"yes\";\r\n      this.opened = \"yes\";\r\n      this.openChange.emit(true);\r\n      this.ref.markForCheck();\r\n\r\n      // 强制更新视图，确保DOM已更新，但弹窗仍然隐藏（因为 positioning = yes）\r\n      this.ref.detectChanges();\r\n\r\n      // 设置位置的函数\r\n      const setPosition = () => {\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          // 位置设置完成后，再显示弹窗（设置 positioning = no）\r\n          // 使用 requestAnimationFrame 确保样式已应用\r\n          requestAnimationFrame(() => {\r\n            this.positioning = \"no\";\r\n            this.ref.markForCheck();\r\n          });\r\n        }\r\n      };\r\n\r\n      // 如果元素已经存在，立即设置位置\r\n      if (this.popupElement && this.popupElement.nativeElement) {\r\n        setPosition();\r\n      } else {\r\n        // 如果元素还不存在，等待渲染后再设置位置\r\n        requestAnimationFrame(() => {\r\n          if (this.popupElement && this.popupElement.nativeElement) {\r\n            setPosition();\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 = (event: Event) => {\r\n    if (this.opened === \"yes\") {\r\n      // 检查滚动是否发生在弹窗内部\r\n      const target = event.target as Element;\r\n      const isScrollOnPopup =\r\n        this.popupElement &&\r\n        this.popupElement.nativeElement &&\r\n        this.popupElement.nativeElement.contains(target);\r\n\r\n      // 如果滚动发生在弹窗内部，不更新位置（避免不必要的计算）\r\n      // 如果滚动发生在外部，更新位置（因为触发元素位置可能改变）\r\n      if (!isScrollOnPopup) {\r\n        this.updatePopupPosition();\r\n      }\r\n    }\r\n  };\r\n\r\n  // 处理弹窗内部的滚轮事件\r\n  onPopupWheel(event: WheelEvent) {\r\n    // 阻止滚轮事件冒泡，避免触发外部滚动处理\r\n    event.stopPropagation();\r\n  }\r\n\r\n  // 处理弹窗内部的滚动事件\r\n  onPopupScroll(event: Event) {\r\n    // 阻止滚动事件冒泡，避免触发外部滚动处理\r\n    event.stopPropagation();\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    // 始终使用用户指定的 position，autoAdjust 只调整 placement（方向），不改变对齐方式\r\n    const 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      // autoAdjust 只调整 placement（上下左右），保持用户指定的 position（对齐方式）\r\n      finalPlacement = optimal.placement;\r\n      // finalPosition 保持用户指定的值，不改变\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    // 始终使用用户指定的 position，autoAdjust 只调整 placement（方向），不改变对齐方式\r\n    const 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      // autoAdjust 只调整 placement（上下左右），保持用户指定的 position（对齐方式）\r\n      finalPlacement = optimal.placement;\r\n      // finalPosition 保持用户指定的值，不改变\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空间不足，也尽量保持用户指定的position\r\n      // 边界约束逻辑会在 calculatePositionCoordinates 中处理位置调整\r\n      // 这样可以保持用户期望的对齐方式\r\n      return {\r\n        placement: preferred,\r\n        position: preferredPosition,\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  // 注意：弹窗使用 position: fixed，所以坐标应该是相对于视口的，不需要加上滚动偏移\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\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        // 弹窗在触发元素上方\r\n        top = triggerRect.top - this.offset;\r\n        translateY = \"-100%\";\r\n        if (position === \"start\") {\r\n          // 左对齐\r\n          left = triggerRect.left;\r\n          translateX = \"0\";\r\n        } else if (position === \"center\") {\r\n          // 居中对齐\r\n          left = triggerRect.left + triggerWidth / 2;\r\n          translateX = \"-50%\";\r\n        } else {\r\n          // 右对齐（end）\r\n          left = triggerRect.right;\r\n          translateX = \"-100%\";\r\n        }\r\n        // 边界约束：确保弹窗不超出视口\r\n        const popupTop = top - popupSize.height;\r\n        if (popupTop < 0) {\r\n          top = popupSize.height + 8; // 距离顶部至少8px\r\n          translateY = \"0\"; // 取消向上偏移\r\n        }\r\n        // 水平边界约束：尽量保持原始对齐方式\r\n        if (position === \"start\") {\r\n          // 左对齐：如果超出右边界，调整到右边界；如果超出左边界，调整到左边界\r\n          if (left + popupSize.width > viewportWidth) {\r\n            left = viewportWidth - popupSize.width - 8;\r\n          }\r\n          if (left < 8) {\r\n            left = 8;\r\n          }\r\n        } else if (position === \"center\") {\r\n          // 居中对齐：计算实际左边界\r\n          const actualLeft = left - popupSize.width / 2;\r\n          if (actualLeft < 8) {\r\n            // 左边界不足，尽量保持居中，但调整到不超出左边界\r\n            left = 8 + popupSize.width / 2;\r\n            translateX = \"-50%\";\r\n          } else if (actualLeft + popupSize.width > viewportWidth - 8) {\r\n            // 右边界不足，尽量保持居中，但调整到不超出右边界\r\n            left = viewportWidth - 8 - popupSize.width / 2;\r\n            translateX = \"-50%\";\r\n          }\r\n        } else {\r\n          // 右对齐（end）：弹窗右边缘与触发元素右边缘对齐\r\n          // left 是弹窗的右边缘位置（因为 translateX = \"-100%\"）\r\n          // 弹窗的左边缘在 left - popupSize.width\r\n          const popupLeftEdge = left - popupSize.width;\r\n          \r\n          // 如果弹窗左边缘超出左边界，调整位置但保持右对齐\r\n          if (popupLeftEdge < 8) {\r\n            left = 8 + popupSize.width;\r\n            translateX = \"-100%\";\r\n          }\r\n          // 如果弹窗右边缘超出右边界，调整位置但保持右对齐\r\n          if (left > viewportWidth - 8) {\r\n            left = viewportWidth - 8;\r\n            translateX = \"-100%\";\r\n          }\r\n        }\r\n        break;\r\n\r\n      case \"bottom\":\r\n        // 弹窗在触发元素下方\r\n        top = triggerRect.bottom + this.offset;\r\n        if (position === \"start\") {\r\n          // 左对齐\r\n          left = triggerRect.left;\r\n          translateX = \"0\";\r\n        } else if (position === \"center\") {\r\n          // 居中对齐\r\n          left = triggerRect.left + triggerWidth / 2;\r\n          translateX = \"-50%\";\r\n        } else {\r\n          // 右对齐（end）\r\n          left = triggerRect.right;\r\n          translateX = \"-100%\";\r\n        }\r\n        // 边界约束：确保弹窗不超出视口\r\n        if (top + popupSize.height > viewportHeight) {\r\n          top = viewportHeight - popupSize.height - 8; // 距离底部至少8px\r\n        }\r\n        // 水平边界约束：尽量保持原始对齐方式\r\n        if (position === \"start\") {\r\n          // 左对齐：如果超出右边界，调整到右边界；如果超出左边界，调整到左边界\r\n          if (left + popupSize.width > viewportWidth) {\r\n            left = viewportWidth - popupSize.width - 8;\r\n          }\r\n          if (left < 8) {\r\n            left = 8;\r\n          }\r\n        } else if (position === \"center\") {\r\n          // 居中对齐：计算实际左边界\r\n          const actualLeft = left - popupSize.width / 2;\r\n          if (actualLeft < 8) {\r\n            // 左边界不足，尽量保持居中，但调整到不超出左边界\r\n            left = 8 + popupSize.width / 2;\r\n            translateX = \"-50%\";\r\n          } else if (actualLeft + popupSize.width > viewportWidth - 8) {\r\n            // 右边界不足，尽量保持居中，但调整到不超出右边界\r\n            left = viewportWidth - 8 - popupSize.width / 2;\r\n            translateX = \"-50%\";\r\n          }\r\n        } else {\r\n          // 右对齐（end）：弹窗右边缘与触发元素右边缘对齐\r\n          // left 是弹窗的右边缘位置（因为 translateX = \"-100%\"）\r\n          // 弹窗的左边缘在 left - popupSize.width\r\n          const popupLeftEdge = left - popupSize.width;\r\n          \r\n          // 优先保持与触发元素右对齐，只在必要时调整\r\n          // 如果弹窗右边缘超出右边界，调整位置但保持右对齐\r\n          if (left > viewportWidth - 8) {\r\n            left = viewportWidth - 8;\r\n            translateX = \"-100%\";\r\n          }\r\n          // 如果弹窗左边缘超出左边界，且弹窗可以完全显示在视口内，才调整位置\r\n          // 这样可以确保在可能的情况下，弹窗始终与触发元素右对齐\r\n          if (popupLeftEdge < 8 && popupSize.width < viewportWidth - 16) {\r\n            // 只有当弹窗可以完全显示在视口内时，才调整位置\r\n            // 否则保持与触发元素右对齐，即使部分超出左边界\r\n            left = 8 + popupSize.width;\r\n            translateX = \"-100%\";\r\n          }\r\n        }\r\n        break;\r\n\r\n      case \"left\":\r\n        // 弹窗在触发元素左侧\r\n        left = triggerRect.left - this.offset;\r\n        translateX = \"-100%\";\r\n        if (position === \"start\") {\r\n          // 上对齐\r\n          top = triggerRect.top;\r\n          translateY = \"0\";\r\n        } else if (position === \"center\") {\r\n          // 居中对齐\r\n          top = triggerRect.top + triggerHeight / 2;\r\n          translateY = \"-50%\";\r\n        } else {\r\n          // 下对齐（end）\r\n          top = triggerRect.bottom;\r\n          translateY = \"-100%\";\r\n        }\r\n        // 边界约束：确保弹窗不超出视口\r\n        const popupLeftSide = left - popupSize.width;\r\n        if (popupLeftSide < 0) {\r\n          left = 8; // 距离左侧至少8px\r\n          translateX = \"0\"; // 取消向左偏移\r\n        }\r\n        // 垂直边界约束：尽量保持原始对齐方式\r\n        if (position === \"start\") {\r\n          // 上对齐：如果超出下边界，调整到下边界；如果超出上边界，调整到上边界\r\n          if (top + popupSize.height > viewportHeight) {\r\n            top = viewportHeight - popupSize.height - 8;\r\n          }\r\n          if (top < 8) {\r\n            top = 8;\r\n          }\r\n        } else if (position === \"center\") {\r\n          // 居中对齐：计算实际上边界\r\n          const actualTop = top - popupSize.height / 2;\r\n          if (actualTop < 8) {\r\n            // 上边界不足，尽量保持居中，但调整到不超出上边界\r\n            top = 8 + popupSize.height / 2;\r\n            translateY = \"-50%\";\r\n          } else if (actualTop + popupSize.height > viewportHeight - 8) {\r\n            // 下边界不足，尽量保持居中，但调整到不超出下边界\r\n            top = viewportHeight - 8 - popupSize.height / 2;\r\n            translateY = \"-50%\";\r\n          }\r\n        } else {\r\n          // 下对齐（end）：如果超出上边界，调整到上边界；如果超出下边界，调整到下边界\r\n          const actualTop = top - popupSize.height;\r\n          if (actualTop < 8) {\r\n            top = 8 + popupSize.height;\r\n            translateY = \"-100%\";\r\n          }\r\n          if (top > viewportHeight - 8) {\r\n            top = viewportHeight - 8;\r\n            translateY = \"-100%\";\r\n          }\r\n        }\r\n        break;\r\n\r\n      case \"right\":\r\n        // 弹窗在触发元素右侧\r\n        left = triggerRect.right + this.offset;\r\n        translateX = \"0\";\r\n        if (position === \"start\") {\r\n          // 上对齐\r\n          top = triggerRect.top;\r\n          translateY = \"0\";\r\n        } else if (position === \"center\") {\r\n          // 居中对齐\r\n          top = triggerRect.top + triggerHeight / 2;\r\n          translateY = \"-50%\";\r\n        } else {\r\n          // 下对齐（end）\r\n          top = triggerRect.bottom;\r\n          translateY = \"-100%\";\r\n        }\r\n        // 边界约束：确保弹窗不超出视口\r\n        if (left + popupSize.width > viewportWidth) {\r\n          left = viewportWidth - popupSize.width - 8; // 距离右侧至少8px\r\n        }\r\n        // 垂直边界约束：尽量保持原始对齐方式\r\n        if (position === \"start\") {\r\n          // 上对齐：如果超出下边界，调整到下边界；如果超出上边界，调整到上边界\r\n          if (top + popupSize.height > viewportHeight) {\r\n            top = viewportHeight - popupSize.height - 8;\r\n          }\r\n          if (top < 8) {\r\n            top = 8;\r\n          }\r\n        } else if (position === \"center\") {\r\n          // 居中对齐：计算实际上边界\r\n          const actualTop = top - popupSize.height / 2;\r\n          if (actualTop < 8) {\r\n            // 上边界不足，尽量保持居中，但调整到不超出上边界\r\n            top = 8 + popupSize.height / 2;\r\n            translateY = \"-50%\";\r\n          } else if (actualTop + popupSize.height > viewportHeight - 8) {\r\n            // 下边界不足，尽量保持居中，但调整到不超出下边界\r\n            top = viewportHeight - 8 - popupSize.height / 2;\r\n            translateY = \"-50%\";\r\n          }\r\n        } else {\r\n          // 下对齐（end）：如果超出上边界，调整到上边界；如果超出下边界，调整到下边界\r\n          const actualTop = top - popupSize.height;\r\n          if (actualTop < 8) {\r\n            top = 8 + popupSize.height;\r\n            translateY = \"-100%\";\r\n          }\r\n          if (top > viewportHeight - 8) {\r\n            top = viewportHeight - 8;\r\n            translateY = \"-100%\";\r\n          }\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","<!-- 触发元素插槽 -->\r\n<ng-container #trigger>\r\n  <ng-content select=\"[trigger]\"></ng-content>\r\n</ng-container>\r\n\r\n<!-- 弹窗内容 -->\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  (wheel)=\"onPopupWheel($event)\"\r\n  (scroll)=\"onPopupScroll($event)\"\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"]}