@skyux/core 8.7.2 → 9.0.0-alpha.1

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 (152) hide show
  1. package/documentation.json +29 -29
  2. package/{esm2020 → esm2022}/lib/modules/adapter-service/adapter.module.mjs +5 -5
  3. package/esm2022/lib/modules/adapter-service/adapter.service.mjs +225 -0
  4. package/esm2022/lib/modules/affix/affix.directive.mjs +116 -0
  5. package/{esm2020 → esm2022}/lib/modules/affix/affix.module.mjs +5 -5
  6. package/esm2022/lib/modules/affix/affix.service.mjs +25 -0
  7. package/esm2022/lib/modules/affix/affixer.mjs +398 -0
  8. package/esm2022/lib/modules/default-input-provider/default-input-provider.mjs +26 -0
  9. package/esm2022/lib/modules/dock/dock-dom-adapter.service.mjs +81 -0
  10. package/esm2022/lib/modules/dock/dock-item.mjs +31 -0
  11. package/esm2022/lib/modules/dock/dock.component.mjs +108 -0
  12. package/{esm2020 → esm2022}/lib/modules/dock/dock.module.mjs +5 -5
  13. package/esm2022/lib/modules/dock/dock.service.mjs +96 -0
  14. package/{esm2020 → esm2022}/lib/modules/dynamic-component/dynamic-component.module.mjs +5 -5
  15. package/esm2022/lib/modules/dynamic-component/dynamic-component.service.mjs +106 -0
  16. package/{esm2020 → esm2022}/lib/modules/format/app-format.mjs +4 -4
  17. package/esm2022/lib/modules/id/id.directive.mjs +31 -0
  18. package/{esm2020 → esm2022}/lib/modules/id/id.module.mjs +5 -5
  19. package/{esm2020 → esm2022}/lib/modules/id/id.service.mjs +4 -4
  20. package/esm2022/lib/modules/layout-host/layout-host.service.mjs +22 -0
  21. package/esm2022/lib/modules/live-announcer/live-announcer.service.mjs +79 -0
  22. package/{esm2020 → esm2022}/lib/modules/log/log.module.mjs +5 -5
  23. package/esm2022/lib/modules/log/log.service.mjs +148 -0
  24. package/{esm2020 → esm2022}/lib/modules/media-query/media-query.module.mjs +5 -5
  25. package/esm2022/lib/modules/media-query/media-query.service.mjs +123 -0
  26. package/{esm2020 → esm2022}/lib/modules/mutation/mutation-observer-service.mjs +4 -4
  27. package/{esm2020 → esm2022}/lib/modules/numeric/numeric.module.mjs +5 -5
  28. package/esm2022/lib/modules/numeric/numeric.pipe.mjs +81 -0
  29. package/esm2022/lib/modules/numeric/numeric.service.mjs +205 -0
  30. package/esm2022/lib/modules/overlay/overlay-adapter.service.mjs +44 -0
  31. package/esm2022/lib/modules/overlay/overlay-instance.mjs +62 -0
  32. package/esm2022/lib/modules/overlay/overlay.component.mjs +193 -0
  33. package/{esm2020 → esm2022}/lib/modules/overlay/overlay.module.mjs +5 -5
  34. package/esm2022/lib/modules/overlay/overlay.service.mjs +124 -0
  35. package/{esm2020 → esm2022}/lib/modules/percent-pipe/percent-pipe.module.mjs +5 -5
  36. package/esm2022/lib/modules/percent-pipe/percent.pipe.mjs +56 -0
  37. package/esm2022/lib/modules/resize-observer/resize-observer-media-query.service.mjs +120 -0
  38. package/esm2022/lib/modules/resize-observer/resize-observer.service.mjs +77 -0
  39. package/esm2022/lib/modules/scrollable-host/scrollable-host.service.mjs +223 -0
  40. package/esm2022/lib/modules/shared/number-format/number-format-utility.mjs +72 -0
  41. package/esm2022/lib/modules/shared/sky-core-resources.module.mjs +50 -0
  42. package/esm2022/lib/modules/title/title.service.mjs +31 -0
  43. package/esm2022/lib/modules/trim/trim.directive.mjs +64 -0
  44. package/{esm2020 → esm2022}/lib/modules/trim/trim.module.mjs +5 -5
  45. package/{esm2020 → esm2022}/lib/modules/ui-config/ui-config.service.mjs +4 -4
  46. package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper-host-options.mjs +4 -4
  47. package/esm2022/lib/modules/viewkeeper/viewkeeper.directive.mjs +131 -0
  48. package/esm2022/lib/modules/viewkeeper/viewkeeper.mjs +283 -0
  49. package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper.module.mjs +5 -5
  50. package/esm2022/lib/modules/viewkeeper/viewkeeper.service.mjs +39 -0
  51. package/{esm2020 → esm2022}/lib/modules/window/window-ref.mjs +4 -4
  52. package/esm2022/testing/core-testing.module.mjs +32 -0
  53. package/esm2022/testing/mock-media-query.service.mjs +46 -0
  54. package/{esm2020 → esm2022}/testing/mock-ui-config.service.mjs +4 -4
  55. package/esm2022/testing/overlay/overlay-harness.mjs +43 -0
  56. package/esm2022/testing/shared/component-harness.mjs +13 -0
  57. package/{fesm2020 → fesm2022}/skyux-core-testing.mjs +28 -34
  58. package/{fesm2020 → fesm2022}/skyux-core-testing.mjs.map +1 -1
  59. package/fesm2022/skyux-core.mjs +3858 -0
  60. package/fesm2022/skyux-core.mjs.map +1 -0
  61. package/lib/modules/affix/affix.directive.d.ts +1 -1
  62. package/lib/modules/viewkeeper/viewkeeper.directive.d.ts +1 -1
  63. package/package.json +14 -22
  64. package/esm2020/lib/modules/adapter-service/adapter.service.mjs +0 -227
  65. package/esm2020/lib/modules/affix/affix.directive.mjs +0 -118
  66. package/esm2020/lib/modules/affix/affix.service.mjs +0 -28
  67. package/esm2020/lib/modules/affix/affixer.mjs +0 -377
  68. package/esm2020/lib/modules/default-input-provider/default-input-provider.mjs +0 -31
  69. package/esm2020/lib/modules/dock/dock-dom-adapter.service.mjs +0 -83
  70. package/esm2020/lib/modules/dock/dock-item.mjs +0 -34
  71. package/esm2020/lib/modules/dock/dock.component.mjs +0 -110
  72. package/esm2020/lib/modules/dock/dock.service.mjs +0 -98
  73. package/esm2020/lib/modules/dynamic-component/dynamic-component.service.mjs +0 -109
  74. package/esm2020/lib/modules/id/id.directive.mjs +0 -34
  75. package/esm2020/lib/modules/layout-host/layout-host.service.mjs +0 -27
  76. package/esm2020/lib/modules/live-announcer/live-announcer.service.mjs +0 -80
  77. package/esm2020/lib/modules/log/log.service.mjs +0 -147
  78. package/esm2020/lib/modules/media-query/media-query.service.mjs +0 -124
  79. package/esm2020/lib/modules/numeric/numeric.pipe.mjs +0 -84
  80. package/esm2020/lib/modules/numeric/numeric.service.mjs +0 -179
  81. package/esm2020/lib/modules/overlay/overlay-adapter.service.mjs +0 -47
  82. package/esm2020/lib/modules/overlay/overlay-instance.mjs +0 -65
  83. package/esm2020/lib/modules/overlay/overlay.component.mjs +0 -192
  84. package/esm2020/lib/modules/overlay/overlay.service.mjs +0 -125
  85. package/esm2020/lib/modules/percent-pipe/percent.pipe.mjs +0 -59
  86. package/esm2020/lib/modules/resize-observer/resize-observer-media-query.service.mjs +0 -121
  87. package/esm2020/lib/modules/resize-observer/resize-observer.service.mjs +0 -79
  88. package/esm2020/lib/modules/scrollable-host/scrollable-host.service.mjs +0 -218
  89. package/esm2020/lib/modules/shared/number-format/number-format-utility.mjs +0 -72
  90. package/esm2020/lib/modules/shared/sky-core-resources.module.mjs +0 -50
  91. package/esm2020/lib/modules/title/title.service.mjs +0 -34
  92. package/esm2020/lib/modules/trim/trim.directive.mjs +0 -65
  93. package/esm2020/lib/modules/viewkeeper/viewkeeper.directive.mjs +0 -131
  94. package/esm2020/lib/modules/viewkeeper/viewkeeper.mjs +0 -276
  95. package/esm2020/lib/modules/viewkeeper/viewkeeper.service.mjs +0 -42
  96. package/esm2020/testing/core-testing.module.mjs +0 -32
  97. package/esm2020/testing/mock-media-query.service.mjs +0 -46
  98. package/esm2020/testing/overlay/overlay-harness.mjs +0 -43
  99. package/esm2020/testing/shared/component-harness.mjs +0 -19
  100. package/fesm2015/skyux-core-testing.mjs +0 -209
  101. package/fesm2015/skyux-core-testing.mjs.map +0 -1
  102. package/fesm2015/skyux-core.mjs +0 -3835
  103. package/fesm2015/skyux-core.mjs.map +0 -1
  104. package/fesm2020/skyux-core.mjs +0 -3825
  105. package/fesm2020/skyux-core.mjs.map +0 -1
  106. /package/{esm2020 → esm2022}/index.mjs +0 -0
  107. /package/{esm2020 → esm2022}/lib/modules/adapter-service/focusable-children-options.mjs +0 -0
  108. /package/{esm2020 → esm2022}/lib/modules/affix/affix-auto-fit-context.mjs +0 -0
  109. /package/{esm2020 → esm2022}/lib/modules/affix/affix-config.mjs +0 -0
  110. /package/{esm2020 → esm2022}/lib/modules/affix/affix-horizontal-alignment.mjs +0 -0
  111. /package/{esm2020 → esm2022}/lib/modules/affix/affix-offset-change.mjs +0 -0
  112. /package/{esm2020 → esm2022}/lib/modules/affix/affix-offset.mjs +0 -0
  113. /package/{esm2020 → esm2022}/lib/modules/affix/affix-placement-change.mjs +0 -0
  114. /package/{esm2020 → esm2022}/lib/modules/affix/affix-placement.mjs +0 -0
  115. /package/{esm2020 → esm2022}/lib/modules/affix/affix-position.mjs +0 -0
  116. /package/{esm2020 → esm2022}/lib/modules/affix/affix-rect.mjs +0 -0
  117. /package/{esm2020 → esm2022}/lib/modules/affix/affix-utils.mjs +0 -0
  118. /package/{esm2020 → esm2022}/lib/modules/affix/affix-vertical-alignment.mjs +0 -0
  119. /package/{esm2020 → esm2022}/lib/modules/affix/dom-utils.mjs +0 -0
  120. /package/{esm2020 → esm2022}/lib/modules/dock/dock-insert-component-config.mjs +0 -0
  121. /package/{esm2020 → esm2022}/lib/modules/dock/dock-item-config.mjs +0 -0
  122. /package/{esm2020 → esm2022}/lib/modules/dock/dock-item-reference.mjs +0 -0
  123. /package/{esm2020 → esm2022}/lib/modules/dock/dock-location.mjs +0 -0
  124. /package/{esm2020 → esm2022}/lib/modules/dock/dock-options.mjs +0 -0
  125. /package/{esm2020 → esm2022}/lib/modules/dock/sort-by-stack-order.mjs +0 -0
  126. /package/{esm2020 → esm2022}/lib/modules/dynamic-component/dynamic-component-location.mjs +0 -0
  127. /package/{esm2020 → esm2022}/lib/modules/dynamic-component/dynamic-component-options.mjs +0 -0
  128. /package/{esm2020 → esm2022}/lib/modules/layout-host/layout-host-for-child-args.mjs +0 -0
  129. /package/{esm2020 → esm2022}/lib/modules/live-announcer/types/live-announcer-args.mjs +0 -0
  130. /package/{esm2020 → esm2022}/lib/modules/live-announcer/types/live-announcer-politeness.mjs +0 -0
  131. /package/{esm2020 → esm2022}/lib/modules/log/types/log-deprecation-args.mjs +0 -0
  132. /package/{esm2020 → esm2022}/lib/modules/log/types/log-level-token.mjs +0 -0
  133. /package/{esm2020 → esm2022}/lib/modules/log/types/log-level.mjs +0 -0
  134. /package/{esm2020 → esm2022}/lib/modules/media-query/media-breakpoints.mjs +0 -0
  135. /package/{esm2020 → esm2022}/lib/modules/media-query/media-query-listener.mjs +0 -0
  136. /package/{esm2020 → esm2022}/lib/modules/numeric/numeric-symbol.mjs +0 -0
  137. /package/{esm2020 → esm2022}/lib/modules/numeric/numeric.options.mjs +0 -0
  138. /package/{esm2020 → esm2022}/lib/modules/overlay/overlay-config.mjs +0 -0
  139. /package/{esm2020 → esm2022}/lib/modules/overlay/overlay-context.mjs +0 -0
  140. /package/{esm2020 → esm2022}/lib/modules/overlay/overlay-position.mjs +0 -0
  141. /package/{esm2020 → esm2022}/lib/modules/stacking-context/stacking-context-token.mjs +0 -0
  142. /package/{esm2020 → esm2022}/lib/modules/stacking-context/stacking-context.mjs +0 -0
  143. /package/{esm2020 → esm2022}/lib/modules/title/set-title-args.mjs +0 -0
  144. /package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper-boundary-info.mjs +0 -0
  145. /package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper-fixed-styles.mjs +0 -0
  146. /package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper-offset.mjs +0 -0
  147. /package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper-options.mjs +0 -0
  148. /package/{esm2020 → esm2022}/skyux-core.mjs +0 -0
  149. /package/{esm2020 → esm2022}/testing/overlay/overlay-harness-filters.mjs +0 -0
  150. /package/{esm2020 → esm2022}/testing/public-api.mjs +0 -0
  151. /package/{esm2020 → esm2022}/testing/shared/harness-filters.mjs +0 -0
  152. /package/{esm2020 → esm2022}/testing/skyux-core-testing.mjs +0 -0
@@ -0,0 +1,398 @@
1
+ import { Subject, fromEvent } from 'rxjs';
2
+ import { SkyAffixAutoFitContext } from './affix-auto-fit-context';
3
+ import { getInversePlacement, getNextPlacement } from './affix-utils';
4
+ import { getElementOffset, getOverflowParents, isOffsetFullyVisibleWithinParent, isOffsetPartiallyVisibleWithinParent, } from './dom-utils';
5
+ const DEFAULT_AFFIX_CONFIG = {
6
+ autoFitContext: SkyAffixAutoFitContext.OverflowParent,
7
+ enableAutoFit: false,
8
+ horizontalAlignment: 'center',
9
+ isSticky: false,
10
+ placement: 'above',
11
+ };
12
+ export class SkyAffixer {
13
+ /**
14
+ * Fires when the affixed element's offset changes.
15
+ */
16
+ get offsetChange() {
17
+ return this.#offsetChangeObs;
18
+ }
19
+ /**
20
+ * Fires when the base element's nearest overflow parent is scrolling. This is useful if you need
21
+ * to perform an additional action during the scroll event but don't want to generate another
22
+ * event listener.
23
+ */
24
+ get overflowScroll() {
25
+ return this.#overflowScrollObs;
26
+ }
27
+ /**
28
+ * Fires when the placement value changes. A `null` value indicates that a suitable
29
+ * placement could not be found.
30
+ */
31
+ get placementChange() {
32
+ return this.#placementChangeObs;
33
+ }
34
+ get #config() {
35
+ return this.#_config;
36
+ }
37
+ set #config(value) {
38
+ const merged = {
39
+ ...DEFAULT_AFFIX_CONFIG,
40
+ ...value,
41
+ };
42
+ // Make sure none of the values are undefined.
43
+ let key;
44
+ for (key in merged) {
45
+ if (merged[key] === undefined) {
46
+ merged[key] = DEFAULT_AFFIX_CONFIG[key];
47
+ }
48
+ }
49
+ this.#_config = merged;
50
+ }
51
+ #affixedElement;
52
+ #baseElement;
53
+ #currentOffset;
54
+ #currentPlacement;
55
+ #offsetChange;
56
+ #offsetChangeObs;
57
+ #overflowParents = [];
58
+ #overflowScroll;
59
+ #overflowScrollObs;
60
+ #placementChange;
61
+ #placementChangeObs;
62
+ #renderer;
63
+ #resizeListener;
64
+ #scrollListeners;
65
+ #_config = DEFAULT_AFFIX_CONFIG;
66
+ constructor(affixedElement, renderer) {
67
+ this.#affixedElement = affixedElement;
68
+ this.#renderer = renderer;
69
+ this.#offsetChange = new Subject();
70
+ this.#overflowScroll = new Subject();
71
+ this.#placementChange = new Subject();
72
+ this.#offsetChangeObs = this.#offsetChange.asObservable();
73
+ this.#overflowScrollObs = this.#overflowScroll.asObservable();
74
+ this.#placementChangeObs = this.#placementChange.asObservable();
75
+ }
76
+ /**
77
+ * Affixes an element to a base element.
78
+ * @param baseElement The base element.
79
+ * @param config Configuration for the affix action.
80
+ */
81
+ affixTo(baseElement, config) {
82
+ this.#reset();
83
+ this.#config = config;
84
+ this.#baseElement = baseElement;
85
+ this.#overflowParents = getOverflowParents(baseElement);
86
+ this.#affix();
87
+ if (this.#config.isSticky) {
88
+ this.#addScrollListeners();
89
+ this.#addResizeListener();
90
+ }
91
+ }
92
+ getConfig() {
93
+ return this.#config;
94
+ }
95
+ /**
96
+ * Re-runs the affix calculation.
97
+ */
98
+ reaffix() {
99
+ // Reset current placement to preferred placement.
100
+ this.#currentPlacement = this.#config.placement;
101
+ this.#affix();
102
+ }
103
+ /**
104
+ * Destroys the affixer.
105
+ */
106
+ destroy() {
107
+ this.#reset();
108
+ this.#placementChange.complete();
109
+ this.#offsetChange.complete();
110
+ this.#overflowScroll.complete();
111
+ }
112
+ #affix() {
113
+ const offset = this.#getOffset();
114
+ if (this.#isNewOffset(offset)) {
115
+ this.#renderer.setStyle(this.#affixedElement, 'top', `${offset.top}px`);
116
+ this.#renderer.setStyle(this.#affixedElement, 'left', `${offset.left}px`);
117
+ this.#offsetChange.next({ offset });
118
+ }
119
+ }
120
+ #getOffset() {
121
+ const parent = this.#getAutoFitContextParent();
122
+ const maxAttempts = 4;
123
+ let attempts = 0;
124
+ let isAffixedElementFullyVisible = false;
125
+ let offset;
126
+ let placement = this.#config.placement;
127
+ const autoFitOverflowOffset = this.#config.autoFitOverflowOffset || {
128
+ bottom: 0,
129
+ left: 0,
130
+ right: 0,
131
+ top: 0,
132
+ };
133
+ if (this.#config.position === 'absolute') {
134
+ autoFitOverflowOffset.top =
135
+ (autoFitOverflowOffset.top || 0) + window.scrollY;
136
+ }
137
+ do {
138
+ offset = this.#getPreferredOffset(placement);
139
+ isAffixedElementFullyVisible = isOffsetFullyVisibleWithinParent(parent, offset, autoFitOverflowOffset);
140
+ if (!this.#config.enableAutoFit) {
141
+ break;
142
+ }
143
+ if (!isAffixedElementFullyVisible) {
144
+ placement =
145
+ attempts % 2 === 0
146
+ ? getInversePlacement(placement)
147
+ : getNextPlacement(placement);
148
+ }
149
+ attempts++;
150
+ } while (!isAffixedElementFullyVisible && attempts < maxAttempts);
151
+ if (isAffixedElementFullyVisible) {
152
+ if (this.#isBaseElementVisible()) {
153
+ this.#notifyPlacementChange(placement);
154
+ }
155
+ else {
156
+ this.#notifyPlacementChange(null);
157
+ }
158
+ return offset;
159
+ }
160
+ if (this.#config.enableAutoFit) {
161
+ this.#notifyPlacementChange(null);
162
+ }
163
+ // No suitable placement was found, so revert to preferred placement.
164
+ return this.#getPreferredOffset(this.#config.placement);
165
+ }
166
+ #getRect(baseElement) {
167
+ const baseDomRect = baseElement.getBoundingClientRect();
168
+ const baseRect = {
169
+ top: baseDomRect.top,
170
+ bottom: baseDomRect.bottom,
171
+ left: baseDomRect.left,
172
+ right: baseDomRect.right,
173
+ width: baseDomRect.width,
174
+ height: baseDomRect.height,
175
+ };
176
+ if (this.#config.position === 'absolute') {
177
+ baseRect.top += window.scrollY;
178
+ baseRect.bottom = baseRect.top + baseDomRect.height;
179
+ }
180
+ return baseRect;
181
+ }
182
+ #getPreferredOffset(placement) {
183
+ if (!this.#baseElement) {
184
+ return { top: 0, left: 0, bottom: 0, right: 0 };
185
+ }
186
+ const affixedRect = this.#getRect(this.#affixedElement);
187
+ const baseRect = this.#getRect(this.#baseElement);
188
+ const horizontalAlignment = this.#config.horizontalAlignment;
189
+ const verticalAlignment = this.#config.verticalAlignment;
190
+ const enableAutoFit = this.#config.enableAutoFit;
191
+ let top;
192
+ let left;
193
+ if (placement === 'above' || placement === 'below') {
194
+ if (placement === 'above') {
195
+ top = baseRect.top - affixedRect.height;
196
+ switch (verticalAlignment) {
197
+ case 'top':
198
+ top = top + affixedRect.height;
199
+ break;
200
+ case 'middle':
201
+ top = top + affixedRect.height / 2;
202
+ break;
203
+ case 'bottom':
204
+ default:
205
+ break;
206
+ }
207
+ }
208
+ else {
209
+ top = baseRect.bottom;
210
+ switch (verticalAlignment) {
211
+ case 'top':
212
+ default:
213
+ break;
214
+ case 'middle':
215
+ top = top - affixedRect.height / 2;
216
+ break;
217
+ case 'bottom':
218
+ top = top - affixedRect.height;
219
+ break;
220
+ }
221
+ }
222
+ switch (horizontalAlignment) {
223
+ case 'left':
224
+ left = baseRect.left;
225
+ break;
226
+ case 'center':
227
+ default:
228
+ left = baseRect.left + baseRect.width / 2 - affixedRect.width / 2;
229
+ break;
230
+ case 'right':
231
+ left = baseRect.right - affixedRect.width;
232
+ break;
233
+ }
234
+ }
235
+ else {
236
+ if (placement === 'left') {
237
+ left = baseRect.left - affixedRect.width;
238
+ }
239
+ else {
240
+ left = baseRect.right;
241
+ }
242
+ switch (verticalAlignment) {
243
+ case 'top':
244
+ top = baseRect.top;
245
+ break;
246
+ case 'middle':
247
+ default:
248
+ top = baseRect.top + baseRect.height / 2 - affixedRect.height / 2;
249
+ break;
250
+ case 'bottom':
251
+ top = baseRect.bottom - affixedRect.height;
252
+ break;
253
+ }
254
+ }
255
+ const offset = { top, left, bottom: 0, right: 0 };
256
+ if (enableAutoFit) {
257
+ const adjustments = this.#adjustOffsetToOverflowParent({ top, left }, placement, this.#baseElement);
258
+ offset.top = adjustments.top;
259
+ offset.left = adjustments.left;
260
+ }
261
+ offset.bottom = offset.top + affixedRect.height;
262
+ offset.right = offset.left + affixedRect.width;
263
+ return offset;
264
+ }
265
+ /**
266
+ * Slightly adjust the offset to fit within the scroll parent's boundaries if
267
+ * the affixed element would otherwise be clipped.
268
+ */
269
+ #adjustOffsetToOverflowParent(offset, placement, baseElement) {
270
+ const parent = this.#getAutoFitContextParent();
271
+ const parentOffset = getElementOffset(parent, this.#config.autoFitOverflowOffset);
272
+ const affixedRect = this.#getRect(this.#affixedElement);
273
+ const baseRect = this.#getRect(baseElement);
274
+ // A pixel value representing the leeway between the edge of the overflow parent and the edge
275
+ // of the base element before it disappears from view.
276
+ // If the visible portion of the base element is less than this pixel value, the auto-fit
277
+ // functionality attempts to find another placement.
278
+ const defaultPixelTolerance = 40;
279
+ let pixelTolerance;
280
+ const originalOffsetTop = offset.top;
281
+ const originalOffsetLeft = offset.left;
282
+ switch (placement) {
283
+ case 'above':
284
+ case 'below':
285
+ // Keep the affixed element within the overflow parent.
286
+ if (offset.left < parentOffset.left) {
287
+ offset.left = parentOffset.left;
288
+ }
289
+ else if (offset.left + affixedRect.width > parentOffset.right) {
290
+ offset.left = parentOffset.right - affixedRect.width;
291
+ }
292
+ // Use a smaller pixel tolerance if the base element width is less than the default.
293
+ pixelTolerance = Math.min(defaultPixelTolerance, baseRect.width);
294
+ // Make sure the affixed element never detaches from the base element.
295
+ if (offset.left + pixelTolerance > baseRect.right ||
296
+ offset.left + affixedRect.width - pixelTolerance < baseRect.left) {
297
+ offset.left = originalOffsetLeft;
298
+ }
299
+ break;
300
+ case 'left':
301
+ case 'right':
302
+ // Keep the affixed element within the overflow parent.
303
+ if (offset.top < parentOffset.top) {
304
+ offset.top = parentOffset.top;
305
+ }
306
+ else if (offset.top + affixedRect.height > parentOffset.bottom) {
307
+ offset.top = parentOffset.bottom - affixedRect.height;
308
+ }
309
+ // Use a smaller pixel tolerance if the base element height is less than the default.
310
+ pixelTolerance = Math.min(defaultPixelTolerance, baseRect.height);
311
+ // Make sure the affixed element never detaches from the base element.
312
+ if (offset.top + pixelTolerance > baseRect.bottom ||
313
+ offset.top + affixedRect.height - pixelTolerance < baseRect.top) {
314
+ offset.top = originalOffsetTop;
315
+ }
316
+ break;
317
+ }
318
+ return offset;
319
+ }
320
+ #getImmediateOverflowParent() {
321
+ return this.#overflowParents[this.#overflowParents.length - 1];
322
+ }
323
+ #getAutoFitContextParent() {
324
+ const bodyElement = this.#overflowParents[0];
325
+ return this.#config.autoFitContext === SkyAffixAutoFitContext.OverflowParent
326
+ ? this.#getImmediateOverflowParent()
327
+ : bodyElement;
328
+ }
329
+ #notifyPlacementChange(placement) {
330
+ if (this.#currentPlacement !== placement) {
331
+ this.#currentPlacement = placement ?? undefined;
332
+ this.#placementChange.next({
333
+ placement,
334
+ });
335
+ }
336
+ }
337
+ #reset() {
338
+ this.#removeScrollListeners();
339
+ this.#removeResizeListener();
340
+ this.#overflowParents = [];
341
+ this.#config =
342
+ this.#baseElement =
343
+ this.#currentPlacement =
344
+ this.#currentOffset =
345
+ undefined;
346
+ }
347
+ #isNewOffset(offset) {
348
+ if (this.#currentOffset === undefined) {
349
+ this.#currentOffset = offset;
350
+ return true;
351
+ }
352
+ if (this.#currentOffset.top === offset.top &&
353
+ this.#currentOffset.left === offset.left) {
354
+ return false;
355
+ }
356
+ this.#currentOffset = offset;
357
+ return true;
358
+ }
359
+ #isBaseElementVisible() {
360
+ if (!this.#baseElement) {
361
+ return false;
362
+ }
363
+ const baseRect = this.#baseElement.getBoundingClientRect();
364
+ return isOffsetPartiallyVisibleWithinParent(this.#getImmediateOverflowParent(), {
365
+ top: baseRect.top,
366
+ left: baseRect.left,
367
+ right: baseRect.right,
368
+ bottom: baseRect.bottom,
369
+ }, this.#config.autoFitOverflowOffset);
370
+ }
371
+ #addScrollListeners() {
372
+ this.#scrollListeners = this.#overflowParents.map((parentElement) => {
373
+ const overflow = parentElement === document.body ? 'window' : parentElement;
374
+ return this.#renderer.listen(overflow, 'scroll', () => {
375
+ this.#affix();
376
+ this.#overflowScroll.next();
377
+ });
378
+ });
379
+ }
380
+ #addResizeListener() {
381
+ this.#resizeListener = fromEvent(window, 'resize').subscribe(() => this.#affix());
382
+ }
383
+ #removeResizeListener() {
384
+ if (this.#resizeListener) {
385
+ this.#resizeListener.unsubscribe();
386
+ this.#resizeListener = undefined;
387
+ }
388
+ }
389
+ #removeScrollListeners() {
390
+ if (this.#scrollListeners) {
391
+ // Remove renderer-generated listeners by calling the listener itself.
392
+ // https://github.com/angular/angular/issues/9368#issuecomment-227199778
393
+ this.#scrollListeners.forEach((listener) => listener());
394
+ this.#scrollListeners = undefined;
395
+ }
396
+ }
397
+ }
398
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"affixer.js","sourceRoot":"","sources":["../../../../../../../../libs/components/core/src/lib/modules/affix/affixer.ts"],"names":[],"mappings":"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","sourcesContent":["import { Renderer2 } from '@angular/core';\n\nimport { Observable, Subject, Subscription, fromEvent } from 'rxjs';\n\nimport { SkyAffixAutoFitContext } from './affix-auto-fit-context';\nimport { SkyAffixConfig } from './affix-config';\nimport { SkyAffixOffset } from './affix-offset';\nimport { SkyAffixOffsetChange } from './affix-offset-change';\nimport { SkyAffixPlacement } from './affix-placement';\nimport { SkyAffixPlacementChange } from './affix-placement-change';\nimport { AffixRect } from './affix-rect';\nimport { getInversePlacement, getNextPlacement } from './affix-utils';\nimport {\n  getElementOffset,\n  getOverflowParents,\n  isOffsetFullyVisibleWithinParent,\n  isOffsetPartiallyVisibleWithinParent,\n} from './dom-utils';\n\n/**\n * Make specific properties required, so that we don't have to\n * do null checks throughout this file.\n */\ntype AffixConfigOrDefaults = SkyAffixConfig &\n  Required<\n    Pick<\n      SkyAffixConfig,\n      | 'autoFitContext'\n      | 'enableAutoFit'\n      | 'horizontalAlignment'\n      | 'isSticky'\n      | 'placement'\n    >\n  >;\n\nconst DEFAULT_AFFIX_CONFIG: AffixConfigOrDefaults = {\n  autoFitContext: SkyAffixAutoFitContext.OverflowParent,\n  enableAutoFit: false,\n  horizontalAlignment: 'center',\n  isSticky: false,\n  placement: 'above',\n};\n\nexport class SkyAffixer {\n  /**\n   * Fires when the affixed element's offset changes.\n   */\n  public get offsetChange(): Observable<SkyAffixOffsetChange> {\n    return this.#offsetChangeObs;\n  }\n\n  /**\n   * Fires when the base element's nearest overflow parent is scrolling. This is useful if you need\n   * to perform an additional action during the scroll event but don't want to generate another\n   * event listener.\n   */\n  public get overflowScroll(): Observable<void> {\n    return this.#overflowScrollObs;\n  }\n\n  /**\n   * Fires when the placement value changes. A `null` value indicates that a suitable\n   * placement could not be found.\n   */\n  public get placementChange(): Observable<SkyAffixPlacementChange> {\n    return this.#placementChangeObs;\n  }\n\n  get #config(): AffixConfigOrDefaults {\n    return this.#_config;\n  }\n\n  set #config(value: SkyAffixConfig | undefined) {\n    const merged: AffixConfigOrDefaults = {\n      ...DEFAULT_AFFIX_CONFIG,\n      ...value,\n    };\n\n    // Make sure none of the values are undefined.\n    let key: keyof typeof merged;\n    for (key in merged) {\n      if (merged[key] === undefined) {\n        (merged as any)[key] = DEFAULT_AFFIX_CONFIG[key];\n      }\n    }\n\n    this.#_config = merged;\n  }\n\n  #affixedElement: HTMLElement;\n\n  #baseElement: HTMLElement | undefined;\n\n  #currentOffset: SkyAffixOffset | undefined;\n\n  #currentPlacement: SkyAffixPlacement | undefined;\n\n  #offsetChange: Subject<SkyAffixOffsetChange>;\n\n  #offsetChangeObs: Observable<SkyAffixOffsetChange>;\n\n  #overflowParents: HTMLElement[] = [];\n\n  #overflowScroll: Subject<void>;\n\n  #overflowScrollObs: Observable<void>;\n\n  #placementChange: Subject<SkyAffixPlacementChange>;\n\n  #placementChangeObs: Observable<SkyAffixPlacementChange>;\n\n  #renderer: Renderer2;\n\n  #resizeListener: Subscription | undefined;\n\n  #scrollListeners: (() => void)[] | undefined;\n\n  #_config: AffixConfigOrDefaults = DEFAULT_AFFIX_CONFIG;\n\n  constructor(affixedElement: HTMLElement, renderer: Renderer2) {\n    this.#affixedElement = affixedElement;\n    this.#renderer = renderer;\n\n    this.#offsetChange = new Subject<SkyAffixOffsetChange>();\n    this.#overflowScroll = new Subject<void>();\n    this.#placementChange = new Subject<SkyAffixPlacementChange>();\n\n    this.#offsetChangeObs = this.#offsetChange.asObservable();\n    this.#overflowScrollObs = this.#overflowScroll.asObservable();\n    this.#placementChangeObs = this.#placementChange.asObservable();\n  }\n\n  /**\n   * Affixes an element to a base element.\n   * @param baseElement The base element.\n   * @param config Configuration for the affix action.\n   */\n  public affixTo(baseElement: HTMLElement, config?: SkyAffixConfig): void {\n    this.#reset();\n\n    this.#config = config;\n    this.#baseElement = baseElement;\n    this.#overflowParents = getOverflowParents(baseElement);\n\n    this.#affix();\n\n    if (this.#config.isSticky) {\n      this.#addScrollListeners();\n      this.#addResizeListener();\n    }\n  }\n\n  public getConfig(): SkyAffixConfig {\n    return this.#config;\n  }\n\n  /**\n   * Re-runs the affix calculation.\n   */\n  public reaffix(): void {\n    // Reset current placement to preferred placement.\n    this.#currentPlacement = this.#config.placement;\n    this.#affix();\n  }\n\n  /**\n   * Destroys the affixer.\n   */\n  public destroy(): void {\n    this.#reset();\n    this.#placementChange.complete();\n    this.#offsetChange.complete();\n    this.#overflowScroll.complete();\n  }\n\n  #affix(): void {\n    const offset = this.#getOffset();\n\n    if (this.#isNewOffset(offset)) {\n      this.#renderer.setStyle(this.#affixedElement, 'top', `${offset.top}px`);\n      this.#renderer.setStyle(this.#affixedElement, 'left', `${offset.left}px`);\n\n      this.#offsetChange.next({ offset });\n    }\n  }\n\n  #getOffset(): SkyAffixOffset {\n    const parent = this.#getAutoFitContextParent();\n\n    const maxAttempts = 4;\n    let attempts = 0;\n\n    let isAffixedElementFullyVisible = false;\n    let offset: Required<SkyAffixOffset>;\n    let placement = this.#config.placement;\n\n    const autoFitOverflowOffset = this.#config.autoFitOverflowOffset || {\n      bottom: 0,\n      left: 0,\n      right: 0,\n      top: 0,\n    };\n\n    if (this.#config.position === 'absolute') {\n      autoFitOverflowOffset.top =\n        (autoFitOverflowOffset.top || 0) + window.scrollY;\n    }\n\n    do {\n      offset = this.#getPreferredOffset(placement);\n      isAffixedElementFullyVisible = isOffsetFullyVisibleWithinParent(\n        parent,\n        offset,\n        autoFitOverflowOffset\n      );\n\n      if (!this.#config.enableAutoFit) {\n        break;\n      }\n\n      if (!isAffixedElementFullyVisible) {\n        placement =\n          attempts % 2 === 0\n            ? getInversePlacement(placement)\n            : getNextPlacement(placement);\n      }\n\n      attempts++;\n    } while (!isAffixedElementFullyVisible && attempts < maxAttempts);\n\n    if (isAffixedElementFullyVisible) {\n      if (this.#isBaseElementVisible()) {\n        this.#notifyPlacementChange(placement);\n      } else {\n        this.#notifyPlacementChange(null);\n      }\n\n      return offset;\n    }\n\n    if (this.#config.enableAutoFit) {\n      this.#notifyPlacementChange(null);\n    }\n\n    // No suitable placement was found, so revert to preferred placement.\n    return this.#getPreferredOffset(this.#config.placement);\n  }\n\n  #getRect(baseElement: HTMLElement): AffixRect {\n    const baseDomRect = baseElement.getBoundingClientRect();\n\n    const baseRect: AffixRect = {\n      top: baseDomRect.top,\n      bottom: baseDomRect.bottom,\n      left: baseDomRect.left,\n      right: baseDomRect.right,\n      width: baseDomRect.width,\n      height: baseDomRect.height,\n    };\n\n    if (this.#config.position === 'absolute') {\n      baseRect.top += window.scrollY;\n      baseRect.bottom = baseRect.top + baseDomRect.height;\n    }\n\n    return baseRect;\n  }\n\n  #getPreferredOffset(placement: SkyAffixPlacement): Required<SkyAffixOffset> {\n    if (!this.#baseElement) {\n      return { top: 0, left: 0, bottom: 0, right: 0 };\n    }\n\n    const affixedRect = this.#getRect(this.#affixedElement);\n    const baseRect = this.#getRect(this.#baseElement);\n\n    const horizontalAlignment = this.#config.horizontalAlignment;\n    const verticalAlignment = this.#config.verticalAlignment;\n    const enableAutoFit = this.#config.enableAutoFit;\n\n    let top: number;\n    let left: number;\n\n    if (placement === 'above' || placement === 'below') {\n      if (placement === 'above') {\n        top = baseRect.top - affixedRect.height;\n\n        switch (verticalAlignment) {\n          case 'top':\n            top = top + affixedRect.height;\n            break;\n          case 'middle':\n            top = top + affixedRect.height / 2;\n            break;\n          case 'bottom':\n          default:\n            break;\n        }\n      } else {\n        top = baseRect.bottom;\n\n        switch (verticalAlignment) {\n          case 'top':\n          default:\n            break;\n          case 'middle':\n            top = top - affixedRect.height / 2;\n            break;\n          case 'bottom':\n            top = top - affixedRect.height;\n            break;\n        }\n      }\n\n      switch (horizontalAlignment) {\n        case 'left':\n          left = baseRect.left;\n          break;\n\n        case 'center':\n        default:\n          left = baseRect.left + baseRect.width / 2 - affixedRect.width / 2;\n          break;\n\n        case 'right':\n          left = baseRect.right - affixedRect.width;\n          break;\n      }\n    } else {\n      if (placement === 'left') {\n        left = baseRect.left - affixedRect.width;\n      } else {\n        left = baseRect.right;\n      }\n\n      switch (verticalAlignment) {\n        case 'top':\n          top = baseRect.top;\n          break;\n\n        case 'middle':\n        default:\n          top = baseRect.top + baseRect.height / 2 - affixedRect.height / 2;\n          break;\n\n        case 'bottom':\n          top = baseRect.bottom - affixedRect.height;\n          break;\n      }\n    }\n\n    const offset: Required<SkyAffixOffset> = { top, left, bottom: 0, right: 0 };\n\n    if (enableAutoFit) {\n      const adjustments = this.#adjustOffsetToOverflowParent(\n        { top, left },\n        placement,\n        this.#baseElement\n      );\n      offset.top = adjustments.top;\n      offset.left = adjustments.left;\n    }\n\n    offset.bottom = offset.top + affixedRect.height;\n    offset.right = offset.left + affixedRect.width;\n\n    return offset;\n  }\n\n  /**\n   * Slightly adjust the offset to fit within the scroll parent's boundaries if\n   * the affixed element would otherwise be clipped.\n   */\n  #adjustOffsetToOverflowParent(\n    offset: { top: number; left: number },\n    placement: SkyAffixPlacement,\n    baseElement: HTMLElement\n  ): { top: number; left: number } {\n    const parent = this.#getAutoFitContextParent();\n    const parentOffset = getElementOffset(\n      parent,\n      this.#config.autoFitOverflowOffset\n    );\n\n    const affixedRect = this.#getRect(this.#affixedElement);\n    const baseRect = this.#getRect(baseElement);\n\n    // A pixel value representing the leeway between the edge of the overflow parent and the edge\n    // of the base element before it disappears from view.\n    // If the visible portion of the base element is less than this pixel value, the auto-fit\n    // functionality attempts to find another placement.\n    const defaultPixelTolerance = 40;\n    let pixelTolerance: number;\n\n    const originalOffsetTop = offset.top;\n    const originalOffsetLeft = offset.left;\n\n    switch (placement) {\n      case 'above':\n      case 'below':\n        // Keep the affixed element within the overflow parent.\n        if (offset.left < parentOffset.left) {\n          offset.left = parentOffset.left;\n        } else if (offset.left + affixedRect.width > parentOffset.right) {\n          offset.left = parentOffset.right - affixedRect.width;\n        }\n\n        // Use a smaller pixel tolerance if the base element width is less than the default.\n        pixelTolerance = Math.min(defaultPixelTolerance, baseRect.width);\n\n        // Make sure the affixed element never detaches from the base element.\n        if (\n          offset.left + pixelTolerance > baseRect.right ||\n          offset.left + affixedRect.width - pixelTolerance < baseRect.left\n        ) {\n          offset.left = originalOffsetLeft;\n        }\n\n        break;\n\n      case 'left':\n      case 'right':\n        // Keep the affixed element within the overflow parent.\n        if (offset.top < parentOffset.top) {\n          offset.top = parentOffset.top;\n        } else if (offset.top + affixedRect.height > parentOffset.bottom) {\n          offset.top = parentOffset.bottom - affixedRect.height;\n        }\n\n        // Use a smaller pixel tolerance if the base element height is less than the default.\n        pixelTolerance = Math.min(defaultPixelTolerance, baseRect.height);\n\n        // Make sure the affixed element never detaches from the base element.\n        if (\n          offset.top + pixelTolerance > baseRect.bottom ||\n          offset.top + affixedRect.height - pixelTolerance < baseRect.top\n        ) {\n          offset.top = originalOffsetTop;\n        }\n\n        break;\n    }\n\n    return offset;\n  }\n\n  #getImmediateOverflowParent(): HTMLElement {\n    return this.#overflowParents[this.#overflowParents.length - 1];\n  }\n\n  #getAutoFitContextParent(): HTMLElement {\n    const bodyElement = this.#overflowParents[0];\n\n    return this.#config.autoFitContext === SkyAffixAutoFitContext.OverflowParent\n      ? this.#getImmediateOverflowParent()\n      : bodyElement;\n  }\n\n  #notifyPlacementChange(placement: SkyAffixPlacement | null): void {\n    if (this.#currentPlacement !== placement) {\n      this.#currentPlacement = placement ?? undefined;\n      this.#placementChange.next({\n        placement,\n      });\n    }\n  }\n\n  #reset(): void {\n    this.#removeScrollListeners();\n    this.#removeResizeListener();\n\n    this.#overflowParents = [];\n\n    this.#config =\n      this.#baseElement =\n      this.#currentPlacement =\n      this.#currentOffset =\n        undefined;\n  }\n\n  #isNewOffset(offset: SkyAffixOffset): boolean {\n    if (this.#currentOffset === undefined) {\n      this.#currentOffset = offset;\n      return true;\n    }\n\n    if (\n      this.#currentOffset.top === offset.top &&\n      this.#currentOffset.left === offset.left\n    ) {\n      return false;\n    }\n\n    this.#currentOffset = offset;\n\n    return true;\n  }\n\n  #isBaseElementVisible(): boolean {\n    if (!this.#baseElement) {\n      return false;\n    }\n\n    const baseRect = this.#baseElement.getBoundingClientRect();\n\n    return isOffsetPartiallyVisibleWithinParent(\n      this.#getImmediateOverflowParent(),\n      {\n        top: baseRect.top,\n        left: baseRect.left,\n        right: baseRect.right,\n        bottom: baseRect.bottom,\n      },\n      this.#config.autoFitOverflowOffset\n    );\n  }\n\n  #addScrollListeners(): void {\n    this.#scrollListeners = this.#overflowParents.map((parentElement) => {\n      const overflow =\n        parentElement === document.body ? 'window' : parentElement;\n      return this.#renderer.listen(overflow, 'scroll', () => {\n        this.#affix();\n        this.#overflowScroll.next();\n      });\n    });\n  }\n\n  #addResizeListener(): void {\n    this.#resizeListener = fromEvent(window, 'resize').subscribe(() =>\n      this.#affix()\n    );\n  }\n\n  #removeResizeListener(): void {\n    if (this.#resizeListener) {\n      this.#resizeListener.unsubscribe();\n      this.#resizeListener = undefined;\n    }\n  }\n\n  #removeScrollListeners(): void {\n    if (this.#scrollListeners) {\n      // Remove renderer-generated listeners by calling the listener itself.\n      // https://github.com/angular/angular/issues/9368#issuecomment-227199778\n      this.#scrollListeners.forEach((listener) => listener());\n      this.#scrollListeners = undefined;\n    }\n  }\n}\n"]}
@@ -0,0 +1,26 @@
1
+ import { ReplaySubject } from 'rxjs';
2
+ /**
3
+ * @internal
4
+ * An API to provide default Angular component input values to child components.
5
+ */
6
+ export class SkyDefaultInputProvider {
7
+ #props = {};
8
+ setValue(componentName, inputName, value) {
9
+ const subject = this.#getSubject(componentName, inputName);
10
+ subject.next(value);
11
+ }
12
+ getValue(componentName, inputName) {
13
+ const test = this.#getSubject(componentName, inputName);
14
+ return test.asObservable();
15
+ }
16
+ #getSubject(componentName, inputName) {
17
+ const componentSubjects = this.#props[componentName] || {};
18
+ const inputSubject = componentSubjects[inputName];
19
+ if (!inputSubject) {
20
+ componentSubjects[inputName] = new ReplaySubject(1);
21
+ this.#props[componentName] = componentSubjects;
22
+ }
23
+ return componentSubjects[inputName];
24
+ }
25
+ }
26
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGVmYXVsdC1pbnB1dC1wcm92aWRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvY29tcG9uZW50cy9jb3JlL3NyYy9saWIvbW9kdWxlcy9kZWZhdWx0LWlucHV0LXByb3ZpZGVyL2RlZmF1bHQtaW5wdXQtcHJvdmlkZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFjLGFBQWEsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUVqRDs7O0dBR0c7QUFDSCxNQUFNLE9BQU8sdUJBQXVCO0lBQ2xDLE1BQU0sR0FBMkQsRUFBRSxDQUFDO0lBRTdELFFBQVEsQ0FBSSxhQUFxQixFQUFFLFNBQWlCLEVBQUUsS0FBUTtRQUNuRSxNQUFNLE9BQU8sR0FBRyxJQUFJLENBQUMsV0FBVyxDQUFDLGFBQWEsRUFBRSxTQUFTLENBQUMsQ0FBQztRQUMzRCxPQUFPLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3RCLENBQUM7SUFFTSxRQUFRLENBQ2IsYUFBcUIsRUFDckIsU0FBaUI7UUFFakIsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxhQUFhLEVBQUUsU0FBUyxDQUFDLENBQUM7UUFDeEQsT0FBTyxJQUFJLENBQUMsWUFBWSxFQUFtQixDQUFDO0lBQzlDLENBQUM7SUFFRCxXQUFXLENBQ1QsYUFBcUIsRUFDckIsU0FBaUI7UUFFakIsTUFBTSxpQkFBaUIsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLGFBQWEsQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUMzRCxNQUFNLFlBQVksR0FBRyxpQkFBaUIsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUVsRCxJQUFJLENBQUMsWUFBWSxFQUFFO1lBQ2pCLGlCQUFpQixDQUFDLFNBQVMsQ0FBQyxHQUFHLElBQUksYUFBYSxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ3BELElBQUksQ0FBQyxNQUFNLENBQUMsYUFBYSxDQUFDLEdBQUcsaUJBQWlCLENBQUM7U0FDaEQ7UUFFRCxPQUFPLGlCQUFpQixDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQ3RDLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE9ic2VydmFibGUsIFJlcGxheVN1YmplY3QgfSBmcm9tICdyeGpzJztcblxuLyoqXG4gKiBAaW50ZXJuYWxcbiAqIEFuIEFQSSB0byBwcm92aWRlIGRlZmF1bHQgQW5ndWxhciBjb21wb25lbnQgaW5wdXQgdmFsdWVzIHRvIGNoaWxkIGNvbXBvbmVudHMuXG4gKi9cbmV4cG9ydCBjbGFzcyBTa3lEZWZhdWx0SW5wdXRQcm92aWRlciB7XG4gICNwcm9wczogUmVjb3JkPHN0cmluZywgUmVjb3JkPHN0cmluZywgUmVwbGF5U3ViamVjdDx1bmtub3duPj4+ID0ge307XG5cbiAgcHVibGljIHNldFZhbHVlPFQ+KGNvbXBvbmVudE5hbWU6IHN0cmluZywgaW5wdXROYW1lOiBzdHJpbmcsIHZhbHVlOiBUKTogdm9pZCB7XG4gICAgY29uc3Qgc3ViamVjdCA9IHRoaXMuI2dldFN1YmplY3QoY29tcG9uZW50TmFtZSwgaW5wdXROYW1lKTtcbiAgICBzdWJqZWN0Lm5leHQodmFsdWUpO1xuICB9XG5cbiAgcHVibGljIGdldFZhbHVlPFQ+KFxuICAgIGNvbXBvbmVudE5hbWU6IHN0cmluZyxcbiAgICBpbnB1dE5hbWU6IHN0cmluZ1xuICApOiBPYnNlcnZhYmxlPFQ+IHwgdW5kZWZpbmVkIHtcbiAgICBjb25zdCB0ZXN0ID0gdGhpcy4jZ2V0U3ViamVjdChjb21wb25lbnROYW1lLCBpbnB1dE5hbWUpO1xuICAgIHJldHVybiB0ZXN0LmFzT2JzZXJ2YWJsZSgpIGFzIE9ic2VydmFibGU8VD47XG4gIH1cblxuICAjZ2V0U3ViamVjdChcbiAgICBjb21wb25lbnROYW1lOiBzdHJpbmcsXG4gICAgaW5wdXROYW1lOiBzdHJpbmdcbiAgKTogUmVwbGF5U3ViamVjdDx1bmtub3duPiB7XG4gICAgY29uc3QgY29tcG9uZW50U3ViamVjdHMgPSB0aGlzLiNwcm9wc1tjb21wb25lbnROYW1lXSB8fCB7fTtcbiAgICBjb25zdCBpbnB1dFN1YmplY3QgPSBjb21wb25lbnRTdWJqZWN0c1tpbnB1dE5hbWVdO1xuXG4gICAgaWYgKCFpbnB1dFN1YmplY3QpIHtcbiAgICAgIGNvbXBvbmVudFN1YmplY3RzW2lucHV0TmFtZV0gPSBuZXcgUmVwbGF5U3ViamVjdCgxKTtcbiAgICAgIHRoaXMuI3Byb3BzW2NvbXBvbmVudE5hbWVdID0gY29tcG9uZW50U3ViamVjdHM7XG4gICAgfVxuXG4gICAgcmV0dXJuIGNvbXBvbmVudFN1YmplY3RzW2lucHV0TmFtZV07XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,81 @@
1
+ import { Injectable, } from '@angular/core';
2
+ import { Subject, fromEvent as observableFromEvent } from 'rxjs';
3
+ import { debounceTime, takeUntil } from 'rxjs/operators';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "../mutation/mutation-observer-service";
6
+ /**
7
+ * @internal
8
+ */
9
+ export class SkyDockDomAdapterService {
10
+ #currentDockHeight;
11
+ #mutationSvc;
12
+ #ngUnsubscribe = new Subject();
13
+ #observer;
14
+ #renderer;
15
+ #styleElement;
16
+ constructor(mutationSvc, rendererFactory) {
17
+ this.#mutationSvc = mutationSvc;
18
+ this.#renderer = rendererFactory.createRenderer(undefined, null);
19
+ }
20
+ ngOnDestroy() {
21
+ if (this.#observer) {
22
+ this.#observer.disconnect();
23
+ }
24
+ this.#ngUnsubscribe.next();
25
+ this.#ngUnsubscribe.complete();
26
+ if (this.#styleElement) {
27
+ this.#destroyStyleElement();
28
+ }
29
+ this.#currentDockHeight = this.#observer = this.#styleElement = undefined;
30
+ }
31
+ setSticky(elementRef) {
32
+ this.#renderer.addClass(elementRef.nativeElement, 'sky-dock-sticky');
33
+ }
34
+ setZIndex(zIndex, elementRef) {
35
+ this.#renderer.setStyle(elementRef.nativeElement, 'z-index', zIndex);
36
+ }
37
+ unbindDock(elementRef) {
38
+ this.#renderer.addClass(elementRef.nativeElement, 'sky-dock-unbound');
39
+ }
40
+ watchDomChanges(elementRef) {
41
+ this.#observer = this.#mutationSvc.create(() => {
42
+ this.#adjustBodyStyles(elementRef);
43
+ });
44
+ this.#observer.observe(elementRef.nativeElement, {
45
+ attributes: true,
46
+ childList: true,
47
+ characterData: true,
48
+ subtree: true,
49
+ });
50
+ observableFromEvent(window, 'resize')
51
+ .pipe(debounceTime(250), takeUntil(this.#ngUnsubscribe))
52
+ .subscribe(() => this.#adjustBodyStyles(elementRef));
53
+ }
54
+ #adjustBodyStyles(elementRef) {
55
+ const dockHeight = elementRef.nativeElement.getBoundingClientRect().height;
56
+ if (dockHeight === this.#currentDockHeight) {
57
+ return;
58
+ }
59
+ // Create a style element to avoid overwriting any existing inline body styles.
60
+ const styleElement = this.#renderer.createElement('style');
61
+ const textNode = this.#renderer.createText(`body { margin-bottom: ${dockHeight}px; }`);
62
+ // Apply a `data-` attribute to make unit testing easier.
63
+ this.#renderer.setAttribute(styleElement, 'data-test-selector', 'sky-layout-dock-bottom-styles');
64
+ this.#renderer.appendChild(styleElement, textNode);
65
+ this.#renderer.appendChild(document.head, styleElement);
66
+ if (this.#styleElement) {
67
+ this.#destroyStyleElement();
68
+ }
69
+ this.#currentDockHeight = dockHeight;
70
+ this.#styleElement = styleElement;
71
+ }
72
+ #destroyStyleElement() {
73
+ this.#renderer.removeChild(document.head, this.#styleElement);
74
+ }
75
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyDockDomAdapterService, deps: [{ token: i1.SkyMutationObserverService }, { token: i0.RendererFactory2 }], target: i0.ɵɵFactoryTarget.Injectable }); }
76
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyDockDomAdapterService }); }
77
+ }
78
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyDockDomAdapterService, decorators: [{
79
+ type: Injectable
80
+ }], ctorParameters: function () { return [{ type: i1.SkyMutationObserverService }, { type: i0.RendererFactory2 }]; } });
81
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG9jay1kb20tYWRhcHRlci5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9jb21wb25lbnRzL2NvcmUvc3JjL2xpYi9tb2R1bGVzL2RvY2svZG9jay1kb20tYWRhcHRlci5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFFTCxVQUFVLEdBSVgsTUFBTSxlQUFlLENBQUM7QUFFdkIsT0FBTyxFQUFFLE9BQU8sRUFBRSxTQUFTLElBQUksbUJBQW1CLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFDakUsT0FBTyxFQUFFLFlBQVksRUFBRSxTQUFTLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQzs7O0FBSXpEOztHQUVHO0FBRUgsTUFBTSxPQUFPLHdCQUF3QjtJQUNuQyxrQkFBa0IsQ0FBcUI7SUFFdkMsWUFBWSxDQUE2QjtJQUV6QyxjQUFjLEdBQUcsSUFBSSxPQUFPLEVBQVEsQ0FBQztJQUVyQyxTQUFTLENBQStCO0lBRXhDLFNBQVMsQ0FBWTtJQUVyQixhQUFhLENBQStCO0lBRTVDLFlBQ0UsV0FBdUMsRUFDdkMsZUFBaUM7UUFFakMsSUFBSSxDQUFDLFlBQVksR0FBRyxXQUFXLENBQUM7UUFDaEMsSUFBSSxDQUFDLFNBQVMsR0FBRyxlQUFlLENBQUMsY0FBYyxDQUFDLFNBQVMsRUFBRSxJQUFJLENBQUMsQ0FBQztJQUNuRSxDQUFDO0lBRU0sV0FBVztRQUNoQixJQUFJLElBQUksQ0FBQyxTQUFTLEVBQUU7WUFDbEIsSUFBSSxDQUFDLFNBQVMsQ0FBQyxVQUFVLEVBQUUsQ0FBQztTQUM3QjtRQUNELElBQUksQ0FBQyxjQUFjLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDM0IsSUFBSSxDQUFDLGNBQWMsQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUUvQixJQUFJLElBQUksQ0FBQyxhQUFhLEVBQUU7WUFDdEIsSUFBSSxDQUFDLG9CQUFvQixFQUFFLENBQUM7U0FDN0I7UUFF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UmVmLFxuICBJbmplY3RhYmxlLFxuICBPbkRlc3Ryb3ksXG4gIFJlbmRlcmVyMixcbiAgUmVuZGVyZXJGYWN0b3J5Mixcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbmltcG9ydCB7IFN1YmplY3QsIGZyb21FdmVudCBhcyBvYnNlcnZhYmxlRnJvbUV2ZW50IH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBkZWJvdW5jZVRpbWUsIHRha2VVbnRpbCB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcblxuaW1wb3J0IHsgU2t5TXV0YXRpb25PYnNlcnZlclNlcnZpY2UgfSBmcm9tICcuLi9tdXRhdGlvbi9tdXRhdGlvbi1vYnNlcnZlci1zZXJ2aWNlJztcblxuLyoqXG4gKiBAaW50ZXJuYWxcbiAqL1xuQEluamVjdGFibGUoKVxuZXhwb3J0IGNsYXNzIFNreURvY2tEb21BZGFwdGVyU2VydmljZSBpbXBsZW1lbnRzIE9uRGVzdHJveSB7XG4gICNjdXJyZW50RG9ja0hlaWdodDogbnVtYmVyIHwgdW5kZWZpbmVkO1xuXG4gICNtdXRhdGlvblN2YzogU2t5TXV0YXRpb25PYnNlcnZlclNlcnZpY2U7XG5cbiAgI25nVW5zdWJzY3JpYmUgPSBuZXcgU3ViamVjdDx2b2lkPigpO1xuXG4gICNvYnNlcnZlcjogTXV0YXRpb25PYnNlcnZlciB8IHVuZGVmaW5lZDtcblxuICAjcmVuZGVyZXI6IFJlbmRlcmVyMjtcblxuICAjc3R5bGVFbGVtZW50OiBIVE1MU3R5bGVFbGVtZW50IHwgdW5kZWZpbmVkO1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIG11dGF0aW9uU3ZjOiBTa3lNdXRhdGlvbk9ic2VydmVyU2VydmljZSxcbiAgICByZW5kZXJlckZhY3Rvcnk6IFJlbmRlcmVyRmFjdG9yeTJcbiAgKSB7XG4gICAgdGhpcy4jbXV0YXRpb25TdmMgPSBtdXRhdGlvblN2YztcbiAgICB0aGlzLiNyZW5kZXJlciA9IHJlbmRlcmVyRmFjdG9yeS5jcmVhdGVSZW5kZXJlcih1bmRlZmluZWQsIG51bGwpO1xuICB9XG5cbiAgcHVibGljIG5nT25EZXN0cm95KCk6IHZvaWQge1xuICAgIGlmICh0aGlzLiNvYnNlcnZlcikge1xuICAgICAgdGhpcy4jb2JzZXJ2ZXIuZGlzY29ubmVjdCgpO1xuICAgIH1cbiAgICB0aGlzLiNuZ1Vuc3Vic2NyaWJlLm5leHQoKTtcbiAgICB0aGlzLiNuZ1Vuc3Vic2NyaWJlLmNvbXBsZXRlKCk7XG5cbiAgICBpZiAodGhpcy4jc3R5bGVFbGVtZW50KSB7XG4gICAgICB0aGlzLiNkZXN0cm95U3R5bGVFbGVtZW50KCk7XG4gICAgfVxuXG4gICAgdGhpcy4jY3VycmVudERvY2tIZWlnaHQgPSB0aGlzLiNvYnNlcnZlciA9IHRoaXMuI3N0eWxlRWxlbWVudCA9IHVuZGVmaW5lZDtcbiAgfVxuXG4gIHB1YmxpYyBzZXRTdGlja3koZWxlbWVudFJlZjogRWxlbWVudFJlZik6IHZvaWQge1xuICAgIHRoaXMuI3JlbmRlcmVyLmFkZENsYXNzKGVsZW1lbnRSZWYubmF0aXZlRWxlbWVudCwgJ3NreS1kb2NrLXN0aWNreScpO1xuICB9XG5cbiAgcHVibGljIHNldFpJbmRleCh6SW5kZXg6IG51bWJlciwgZWxlbWVudFJlZjogRWxlbWVudFJlZik6IHZvaWQge1xuICAgIHRoaXMuI3JlbmRlcmVyLnNldFN0eWxlKGVsZW1lbnRSZWYubmF0aXZlRWxlbWVudCwgJ3otaW5kZXgnLCB6SW5kZXgpO1xuICB9XG5cbiAgcHVibGljIHVuYmluZERvY2soZWxlbWVudFJlZjogRWxlbWVudFJlZik6IHZvaWQge1xuICAgIHRoaXMuI3JlbmRlcmVyLmFkZENsYXNzKGVsZW1lbnRSZWYubmF0aXZlRWxlbWVudCwgJ3NreS1kb2NrLXVuYm91bmQnKTtcbiAgfVxuXG4gIHB1YmxpYyB3YXRjaERvbUNoYW5nZXMoZWxlbWVudFJlZjogRWxlbWVudFJlZik6IHZvaWQge1xuICAgIHRoaXMuI29ic2VydmVyID0gdGhpcy4jbXV0YXRpb25TdmMuY3JlYXRlKCgpID0+IHtcbiAgICAgIHRoaXMuI2FkanVzdEJvZHlTdHlsZXMoZWxlbWVudFJlZik7XG4gICAgfSk7XG5cbiAgICB0aGlzLiNvYnNlcnZlci5vYnNlcnZlKGVsZW1lbnRSZWYubmF0aXZlRWxlbWVudCwge1xuICAgICAgYXR0cmlidXRlczogdHJ1ZSxcbiAgICAgIGNoaWxkTGlzdDogdHJ1ZSxcbiAgICAgIGNoYXJhY3RlckRhdGE6IHRydWUsXG4gICAgICBzdWJ0cmVlOiB0cnVlLFxuICAgIH0pO1xuXG4gICAgb2JzZXJ2YWJsZUZyb21FdmVudCh3aW5kb3csICdyZXNpemUnKVxuICAgICAgLnBpcGUoZGVib3VuY2VUaW1lKDI1MCksIHRha2VVbnRpbCh0aGlzLiNuZ1Vuc3Vic2NyaWJlKSlcbiAgICAgIC5zdWJzY3JpYmUoKCkgPT4gdGhpcy4jYWRqdXN0Qm9keVN0eWxlcyhlbGVtZW50UmVmKSk7XG4gIH1cblxuICAjYWRqdXN0Qm9keVN0eWxlcyhlbGVtZW50UmVmOiBFbGVtZW50UmVmKTogdm9pZCB7XG4gICAgY29uc3QgZG9ja0hlaWdodCA9IGVsZW1lbnRSZWYubmF0aXZlRWxlbWVudC5nZXRCb3VuZGluZ0NsaWVudFJlY3QoKS5oZWlnaHQ7XG4gICAgaWYgKGRvY2tIZWlnaHQgPT09IHRoaXMuI2N1cnJlbnREb2NrSGVpZ2h0KSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgLy8gQ3JlYXRlIGEgc3R5bGUgZWxlbWVudCB0byBhdm9pZCBvdmVyd3JpdGluZyBhbnkgZXhpc3RpbmcgaW5saW5lIGJvZHkgc3R5bGVzLlxuICAgIGNvbnN0IHN0eWxlRWxlbWVudCA9IHRoaXMuI3JlbmRlcmVyLmNyZWF0ZUVsZW1lbnQoJ3N0eWxlJyk7XG4gICAgY29uc3QgdGV4dE5vZGUgPSB0aGlzLiNyZW5kZXJlci5jcmVhdGVUZXh0KFxuICAgICAgYGJvZHkgeyBtYXJnaW4tYm90dG9tOiAke2RvY2tIZWlnaHR9cHg7IH1gXG4gICAgKTtcblxuICAgIC8vIEFwcGx5IGEgYGRhdGEtYCBhdHRyaWJ1dGUgdG8gbWFrZSB1bml0IHRlc3RpbmcgZWFzaWVyLlxuICAgIHRoaXMuI3JlbmRlcmVyLnNldEF0dHJpYnV0ZShcbiAgICAgIHN0eWxlRWxlbWVudCxcbiAgICAgICdkYXRhLXRlc3Qtc2VsZWN0b3InLFxuICAgICAgJ3NreS1sYXlvdXQtZG9jay1ib3R0b20tc3R5bGVzJ1xuICAgICk7XG5cbiAgICB0aGlzLiNyZW5kZXJlci5hcHBlbmRDaGlsZChzdHlsZUVsZW1lbnQsIHRleHROb2RlKTtcbiAgICB0aGlzLiNyZW5kZXJlci5hcHBlbmRDaGlsZChkb2N1bWVudC5oZWFkLCBzdHlsZUVsZW1lbnQpO1xuXG4gICAgaWYgKHRoaXMuI3N0eWxlRWxlbWVudCkge1xuICAgICAgdGhpcy4jZGVzdHJveVN0eWxlRWxlbWVudCgpO1xuICAgIH1cblxuICAgIHRoaXMuI2N1cnJlbnREb2NrSGVpZ2h0ID0gZG9ja0hlaWdodDtcbiAgICB0aGlzLiNzdHlsZUVsZW1lbnQgPSBzdHlsZUVsZW1lbnQ7XG4gIH1cblxuICAjZGVzdHJveVN0eWxlRWxlbWVudCgpOiB2b2lkIHtcbiAgICB0aGlzLiNyZW5kZXJlci5yZW1vdmVDaGlsZChkb2N1bWVudC5oZWFkLCB0aGlzLiNzdHlsZUVsZW1lbnQpO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,31 @@
1
+ import { Subject } from 'rxjs';
2
+ /**
3
+ * Represents a single item added to the dock.
4
+ */
5
+ export class SkyDockItem {
6
+ /**
7
+ * An event that emits when the item is removed from the dock.
8
+ */
9
+ get destroyed() {
10
+ return this.#destroyedObs;
11
+ }
12
+ #destroyed = new Subject();
13
+ #destroyedObs;
14
+ /**
15
+ * @param componentInstance The item's component instance.
16
+ * @param stackOrder The assigned stack order of the docked item.
17
+ */
18
+ constructor(componentInstance, stackOrder) {
19
+ this.componentInstance = componentInstance;
20
+ this.stackOrder = stackOrder;
21
+ this.#destroyedObs = this.#destroyed.asObservable();
22
+ }
23
+ /**
24
+ * Removes the item from the dock.
25
+ */
26
+ destroy() {
27
+ this.#destroyed.next();
28
+ this.#destroyed.complete();
29
+ }
30
+ }
31
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG9jay1pdGVtLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9jb21wb25lbnRzL2NvcmUvc3JjL2xpYi9tb2R1bGVzL2RvY2svZG9jay1pdGVtLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBYyxPQUFPLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFFM0M7O0dBRUc7QUFDSCxNQUFNLE9BQU8sV0FBVztJQUN0Qjs7T0FFRztJQUNILElBQVcsU0FBUztRQUNsQixPQUFPLElBQUksQ0FBQyxhQUFhLENBQUM7SUFDNUIsQ0FBQztJQUVELFVBQVUsR0FBRyxJQUFJLE9BQU8sRUFBUSxDQUFDO0lBRWpDLGFBQWEsQ0FBbUI7SUFFaEM7OztPQUdHO0lBQ0gsWUFDa0IsaUJBQW9CLEVBQ3BCLFVBQWtCO1FBRGxCLHNCQUFpQixHQUFqQixpQkFBaUIsQ0FBRztRQUNwQixlQUFVLEdBQVYsVUFBVSxDQUFRO1FBRWxDLElBQUksQ0FBQyxhQUFhLEdBQUcsSUFBSSxDQUFDLFVBQVUsQ0FBQyxZQUFZLEVBQUUsQ0FBQztJQUN0RCxDQUFDO0lBRUQ7O09BRUc7SUFDSSxPQUFPO1FBQ1osSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUN2QixJQUFJLENBQUMsVUFBVSxDQUFDLFFBQVEsRUFBRSxDQUFDO0lBQzdCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE9ic2VydmFibGUsIFN1YmplY3QgfSBmcm9tICdyeGpzJztcblxuLyoqXG4gKiBSZXByZXNlbnRzIGEgc2luZ2xlIGl0ZW0gYWRkZWQgdG8gdGhlIGRvY2suXG4gKi9cbmV4cG9ydCBjbGFzcyBTa3lEb2NrSXRlbTxUPiB7XG4gIC8qKlxuICAgKiBBbiBldmVudCB0aGF0IGVtaXRzIHdoZW4gdGhlIGl0ZW0gaXMgcmVtb3ZlZCBmcm9tIHRoZSBkb2NrLlxuICAgKi9cbiAgcHVibGljIGdldCBkZXN0cm95ZWQoKTogT2JzZXJ2YWJsZTx2b2lkPiB7XG4gICAgcmV0dXJuIHRoaXMuI2Rlc3Ryb3llZE9icztcbiAgfVxuXG4gICNkZXN0cm95ZWQgPSBuZXcgU3ViamVjdDx2b2lkPigpO1xuXG4gICNkZXN0cm95ZWRPYnM6IE9ic2VydmFibGU8dm9pZD47XG5cbiAgLyoqXG4gICAqIEBwYXJhbSBjb21wb25lbnRJbnN0YW5jZSBUaGUgaXRlbSdzIGNvbXBvbmVudCBpbnN0YW5jZS5cbiAgICogQHBhcmFtIHN0YWNrT3JkZXIgVGhlIGFzc2lnbmVkIHN0YWNrIG9yZGVyIG9mIHRoZSBkb2NrZWQgaXRlbS5cbiAgICovXG4gIGNvbnN0cnVjdG9yKFxuICAgIHB1YmxpYyByZWFkb25seSBjb21wb25lbnRJbnN0YW5jZTogVCxcbiAgICBwdWJsaWMgcmVhZG9ubHkgc3RhY2tPcmRlcjogbnVtYmVyXG4gICkge1xuICAgIHRoaXMuI2Rlc3Ryb3llZE9icyA9IHRoaXMuI2Rlc3Ryb3llZC5hc09ic2VydmFibGUoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZW1vdmVzIHRoZSBpdGVtIGZyb20gdGhlIGRvY2suXG4gICAqL1xuICBwdWJsaWMgZGVzdHJveSgpOiB2b2lkIHtcbiAgICB0aGlzLiNkZXN0cm95ZWQubmV4dCgpO1xuICAgIHRoaXMuI2Rlc3Ryb3llZC5jb21wbGV0ZSgpO1xuICB9XG59XG4iXX0=