@skyux/core 6.16.0 → 7.0.0-alpha.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/documentation.json +1049 -899
- package/esm2020/lib/modules/adapter-service/adapter.module.mjs +4 -4
- package/esm2020/lib/modules/adapter-service/adapter.service.mjs +24 -25
- package/esm2020/lib/modules/affix/affix-offset.mjs +1 -1
- package/esm2020/lib/modules/affix/affix.directive.mjs +41 -26
- package/esm2020/lib/modules/affix/affix.module.mjs +5 -5
- package/esm2020/lib/modules/affix/affix.service.mjs +3 -3
- package/esm2020/lib/modules/affix/affixer.mjs +266 -260
- package/esm2020/lib/modules/affix/dom-utils.mjs +7 -7
- package/esm2020/lib/modules/dock/dock-dom-adapter.service.mjs +51 -47
- package/esm2020/lib/modules/dock/dock-item.mjs +10 -5
- package/esm2020/lib/modules/dock/dock.component.mjs +58 -34
- package/esm2020/lib/modules/dock/dock.module.mjs +5 -5
- package/esm2020/lib/modules/dock/dock.service.mjs +43 -39
- package/esm2020/lib/modules/dynamic-component/dynamic-component.module.mjs +4 -4
- package/esm2020/lib/modules/dynamic-component/dynamic-component.service.mjs +40 -25
- package/esm2020/lib/modules/format/app-format.mjs +3 -3
- package/esm2020/lib/modules/id/id.directive.mjs +3 -3
- package/esm2020/lib/modules/id/id.module.mjs +4 -4
- package/esm2020/lib/modules/id/id.service.mjs +3 -3
- package/esm2020/lib/modules/log/log.module.mjs +4 -4
- package/esm2020/lib/modules/log/log.service.mjs +41 -41
- package/esm2020/lib/modules/media-query/media-query.module.mjs +4 -4
- package/esm2020/lib/modules/media-query/media-query.service.mjs +53 -50
- package/esm2020/lib/modules/mutation/mutation-observer-service.mjs +6 -4
- package/esm2020/lib/modules/numeric/numeric.module.mjs +5 -5
- package/esm2020/lib/modules/numeric/numeric.pipe.mjs +29 -22
- package/esm2020/lib/modules/numeric/numeric.service.mjs +87 -102
- package/esm2020/lib/modules/overlay/overlay-adapter.service.mjs +26 -20
- package/esm2020/lib/modules/overlay/overlay-instance.mjs +18 -9
- package/esm2020/lib/modules/overlay/overlay.component.mjs +82 -58
- package/esm2020/lib/modules/overlay/overlay.module.mjs +5 -5
- package/esm2020/lib/modules/overlay/overlay.service.mjs +58 -52
- package/esm2020/lib/modules/percent-pipe/percent-pipe.module.mjs +5 -5
- package/esm2020/lib/modules/percent-pipe/percent.pipe.mjs +35 -24
- package/esm2020/lib/modules/resize-observer/resize-observer-media-query.service.mjs +41 -49
- package/esm2020/lib/modules/resize-observer/resize-observer.service.mjs +3 -3
- package/esm2020/lib/modules/scrollable-host/scrollable-host.service.mjs +69 -70
- package/esm2020/lib/modules/shared/sky-core-resources.module.mjs +4 -4
- package/esm2020/lib/modules/title/title.service.mjs +11 -7
- package/esm2020/lib/modules/trim/trim.directive.mjs +13 -14
- package/esm2020/lib/modules/trim/trim.module.mjs +4 -4
- package/esm2020/lib/modules/ui-config/ui-config.service.mjs +3 -3
- package/esm2020/lib/modules/viewkeeper/viewkeeper-boundary-info.mjs +1 -1
- package/esm2020/lib/modules/viewkeeper/viewkeeper-host-options.mjs +3 -3
- package/esm2020/lib/modules/viewkeeper/viewkeeper.directive.mjs +78 -63
- package/esm2020/lib/modules/viewkeeper/viewkeeper.mjs +166 -156
- package/esm2020/lib/modules/viewkeeper/viewkeeper.module.mjs +4 -4
- package/esm2020/lib/modules/viewkeeper/viewkeeper.service.mjs +10 -6
- package/esm2020/lib/modules/window/window-ref.mjs +3 -3
- package/esm2020/testing/core-testing.module.mjs +8 -5
- package/esm2020/testing/mock-media-query.service.mjs +7 -4
- package/esm2020/testing/mock-ui-config.service.mjs +7 -4
- package/esm2020/testing/overlay/overlay-harness-filters.mjs +1 -1
- package/esm2020/testing/overlay/overlay-harness.mjs +2 -1
- package/esm2020/testing/shared/component-harness.mjs +4 -1
- package/esm2020/testing/shared/harness-filters.mjs +1 -1
- package/fesm2015/skyux-core-testing.mjs +23 -10
- package/fesm2015/skyux-core-testing.mjs.map +1 -1
- package/fesm2015/skyux-core.mjs +1416 -1291
- package/fesm2015/skyux-core.mjs.map +1 -1
- package/fesm2020/skyux-core-testing.mjs +23 -10
- package/fesm2020/skyux-core-testing.mjs.map +1 -1
- package/fesm2020/skyux-core.mjs +1418 -1290
- package/fesm2020/skyux-core.mjs.map +1 -1
- package/lib/modules/adapter-service/adapter.service.d.ts +3 -5
- package/lib/modules/affix/affix-offset.d.ts +1 -1
- package/lib/modules/affix/affix.directive.d.ts +13 -14
- package/lib/modules/affix/affixer.d.ts +2 -34
- package/lib/modules/affix/dom-utils.d.ts +3 -3
- package/lib/modules/dock/dock-dom-adapter.service.d.ts +2 -9
- package/lib/modules/dock/dock-item.d.ts +1 -1
- package/lib/modules/dock/dock.component.d.ts +3 -11
- package/lib/modules/dock/dock.service.d.ts +2 -5
- package/lib/modules/dynamic-component/dynamic-component.service.d.ts +2 -7
- package/lib/modules/id/id.directive.d.ts +1 -1
- package/lib/modules/log/log.service.d.ts +2 -6
- package/lib/modules/media-query/media-query.service.d.ts +1 -8
- package/lib/modules/numeric/numeric.pipe.d.ts +3 -9
- package/lib/modules/numeric/numeric.service.d.ts +3 -36
- package/lib/modules/overlay/overlay-adapter.service.d.ts +1 -3
- package/lib/modules/overlay/overlay-instance.d.ts +1 -2
- package/lib/modules/overlay/overlay.component.d.ts +2 -15
- package/lib/modules/overlay/overlay.service.d.ts +1 -7
- package/lib/modules/percent-pipe/percent.pipe.d.ts +3 -10
- package/lib/modules/resize-observer/resize-observer-media-query.service.d.ts +2 -7
- package/lib/modules/scrollable-host/scrollable-host.service.d.ts +2 -11
- package/lib/modules/title/title.service.d.ts +2 -2
- package/lib/modules/trim/trim.directive.d.ts +1 -3
- package/lib/modules/viewkeeper/viewkeeper-boundary-info.d.ts +1 -1
- package/lib/modules/viewkeeper/viewkeeper.d.ts +2 -24
- package/lib/modules/viewkeeper/viewkeeper.directive.d.ts +5 -17
- package/lib/modules/viewkeeper/viewkeeper.service.d.ts +1 -1
- package/package.json +10 -10
- package/testing/core-testing.module.d.ts +3 -0
- package/testing/{skyux-core-testing.d.ts → index.d.ts} +0 -0
- package/testing/mock-media-query.service.d.ts +3 -0
- package/testing/mock-ui-config.service.d.ts +3 -0
- package/testing/overlay/overlay-harness-filters.d.ts +1 -0
- package/testing/overlay/overlay-harness.d.ts +1 -0
- package/testing/shared/component-harness.d.ts +3 -0
- package/testing/shared/harness-filters.d.ts +1 -0
- package/skyux-core.d.ts +0 -5
- package/testing/package.json +0 -10
@@ -1,3 +1,5 @@
|
|
1
|
+
var _SkyAffixer_instances, _SkyAffixer_config_get, _SkyAffixer_config_set, _SkyAffixer_affixedElement, _SkyAffixer_baseElement, _SkyAffixer_currentOffset, _SkyAffixer_currentPlacement, _SkyAffixer_offsetChange, _SkyAffixer_offsetChangeObs, _SkyAffixer_overflowParents, _SkyAffixer_overflowScroll, _SkyAffixer_overflowScrollObs, _SkyAffixer_placementChange, _SkyAffixer_placementChangeObs, _SkyAffixer_renderer, _SkyAffixer_resizeListener, _SkyAffixer_scrollListeners, _SkyAffixer__config, _SkyAffixer_affix, _SkyAffixer_getOffset, _SkyAffixer_getPreferredOffset, _SkyAffixer_adjustOffsetToOverflowParent, _SkyAffixer_getImmediateOverflowParent, _SkyAffixer_getAutoFitContextParent, _SkyAffixer_notifyPlacementChange, _SkyAffixer_reset, _SkyAffixer_isNewOffset, _SkyAffixer_isBaseElementVisible, _SkyAffixer_addScrollListeners, _SkyAffixer_addResizeListener, _SkyAffixer_removeResizeListener, _SkyAffixer_removeScrollListeners;
|
2
|
+
import { __classPrivateFieldGet, __classPrivateFieldSet } from "tslib";
|
1
3
|
import { Subject, fromEvent } from 'rxjs';
|
2
4
|
import { SkyAffixAutoFitContext } from './affix-auto-fit-context';
|
3
5
|
import { getInversePlacement, getNextPlacement } from './affix-utils';
|
@@ -11,17 +13,36 @@ const DEFAULT_AFFIX_CONFIG = {
|
|
11
13
|
};
|
12
14
|
export class SkyAffixer {
|
13
15
|
constructor(affixedElement, renderer) {
|
14
|
-
this
|
15
|
-
this
|
16
|
-
this
|
17
|
-
this
|
18
|
-
this
|
16
|
+
_SkyAffixer_instances.add(this);
|
17
|
+
_SkyAffixer_affixedElement.set(this, void 0);
|
18
|
+
_SkyAffixer_baseElement.set(this, void 0);
|
19
|
+
_SkyAffixer_currentOffset.set(this, void 0);
|
20
|
+
_SkyAffixer_currentPlacement.set(this, void 0);
|
21
|
+
_SkyAffixer_offsetChange.set(this, void 0);
|
22
|
+
_SkyAffixer_offsetChangeObs.set(this, void 0);
|
23
|
+
_SkyAffixer_overflowParents.set(this, []);
|
24
|
+
_SkyAffixer_overflowScroll.set(this, void 0);
|
25
|
+
_SkyAffixer_overflowScrollObs.set(this, void 0);
|
26
|
+
_SkyAffixer_placementChange.set(this, void 0);
|
27
|
+
_SkyAffixer_placementChangeObs.set(this, void 0);
|
28
|
+
_SkyAffixer_renderer.set(this, void 0);
|
29
|
+
_SkyAffixer_resizeListener.set(this, void 0);
|
30
|
+
_SkyAffixer_scrollListeners.set(this, void 0);
|
31
|
+
_SkyAffixer__config.set(this, DEFAULT_AFFIX_CONFIG);
|
32
|
+
__classPrivateFieldSet(this, _SkyAffixer_affixedElement, affixedElement, "f");
|
33
|
+
__classPrivateFieldSet(this, _SkyAffixer_renderer, renderer, "f");
|
34
|
+
__classPrivateFieldSet(this, _SkyAffixer_offsetChange, new Subject(), "f");
|
35
|
+
__classPrivateFieldSet(this, _SkyAffixer_overflowScroll, new Subject(), "f");
|
36
|
+
__classPrivateFieldSet(this, _SkyAffixer_placementChange, new Subject(), "f");
|
37
|
+
__classPrivateFieldSet(this, _SkyAffixer_offsetChangeObs, __classPrivateFieldGet(this, _SkyAffixer_offsetChange, "f").asObservable(), "f");
|
38
|
+
__classPrivateFieldSet(this, _SkyAffixer_overflowScrollObs, __classPrivateFieldGet(this, _SkyAffixer_overflowScroll, "f").asObservable(), "f");
|
39
|
+
__classPrivateFieldSet(this, _SkyAffixer_placementChangeObs, __classPrivateFieldGet(this, _SkyAffixer_placementChange, "f").asObservable(), "f");
|
19
40
|
}
|
20
41
|
/**
|
21
42
|
* Fires when the affixed element's offset changes.
|
22
43
|
*/
|
23
44
|
get offsetChange() {
|
24
|
-
return this
|
45
|
+
return __classPrivateFieldGet(this, _SkyAffixer_offsetChangeObs, "f");
|
25
46
|
}
|
26
47
|
/**
|
27
48
|
* Fires when the base element's nearest overflow parent is scrolling. This is useful if you need
|
@@ -29,27 +50,14 @@ export class SkyAffixer {
|
|
29
50
|
* event listener.
|
30
51
|
*/
|
31
52
|
get overflowScroll() {
|
32
|
-
return this
|
53
|
+
return __classPrivateFieldGet(this, _SkyAffixer_overflowScrollObs, "f");
|
33
54
|
}
|
34
55
|
/**
|
35
56
|
* Fires when the placement value changes. A `null` value indicates that a suitable
|
36
57
|
* placement could not be found.
|
37
58
|
*/
|
38
59
|
get placementChange() {
|
39
|
-
return this
|
40
|
-
}
|
41
|
-
get config() {
|
42
|
-
return this._config;
|
43
|
-
}
|
44
|
-
set config(value) {
|
45
|
-
const merged = { ...DEFAULT_AFFIX_CONFIG, ...value };
|
46
|
-
// Make sure none of the values are undefined.
|
47
|
-
Object.keys(merged).forEach((k) => {
|
48
|
-
if (merged[k] === undefined) {
|
49
|
-
merged[k] = DEFAULT_AFFIX_CONFIG[k];
|
50
|
-
}
|
51
|
-
});
|
52
|
-
this._config = merged;
|
60
|
+
return __classPrivateFieldGet(this, _SkyAffixer_placementChangeObs, "f");
|
53
61
|
}
|
54
62
|
/**
|
55
63
|
* Affixes an element to a base element.
|
@@ -57,290 +65,288 @@ export class SkyAffixer {
|
|
57
65
|
* @param config Configuration for the affix action.
|
58
66
|
*/
|
59
67
|
affixTo(baseElement, config) {
|
60
|
-
this.
|
61
|
-
this
|
62
|
-
this
|
63
|
-
this
|
64
|
-
this.
|
65
|
-
if (this.
|
66
|
-
this.
|
67
|
-
this.
|
68
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_reset).call(this);
|
69
|
+
__classPrivateFieldSet(this, _SkyAffixer_instances, config, "a", _SkyAffixer_config_set);
|
70
|
+
__classPrivateFieldSet(this, _SkyAffixer_baseElement, baseElement, "f");
|
71
|
+
__classPrivateFieldSet(this, _SkyAffixer_overflowParents, getOverflowParents(baseElement), "f");
|
72
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_affix).call(this);
|
73
|
+
if (__classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).isSticky) {
|
74
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_addScrollListeners).call(this);
|
75
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_addResizeListener).call(this);
|
68
76
|
}
|
69
77
|
}
|
78
|
+
getConfig() {
|
79
|
+
return __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get);
|
80
|
+
}
|
70
81
|
/**
|
71
82
|
* Re-runs the affix calculation.
|
72
83
|
*/
|
73
84
|
reaffix() {
|
74
85
|
// Reset current placement to preferred placement.
|
75
|
-
this
|
76
|
-
this.
|
86
|
+
__classPrivateFieldSet(this, _SkyAffixer_currentPlacement, __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).placement, "f");
|
87
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_affix).call(this);
|
77
88
|
}
|
78
89
|
/**
|
79
90
|
* Destroys the affixer.
|
80
91
|
*/
|
81
92
|
destroy() {
|
82
|
-
this.
|
83
|
-
this.
|
84
|
-
this.
|
85
|
-
this.
|
86
|
-
this._offsetChange =
|
87
|
-
this._placementChange =
|
88
|
-
this._overflowScroll =
|
89
|
-
undefined;
|
93
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_reset).call(this);
|
94
|
+
__classPrivateFieldGet(this, _SkyAffixer_placementChange, "f").complete();
|
95
|
+
__classPrivateFieldGet(this, _SkyAffixer_offsetChange, "f").complete();
|
96
|
+
__classPrivateFieldGet(this, _SkyAffixer_overflowScroll, "f").complete();
|
90
97
|
}
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
98
|
+
}
|
99
|
+
_SkyAffixer_affixedElement = new WeakMap(), _SkyAffixer_baseElement = new WeakMap(), _SkyAffixer_currentOffset = new WeakMap(), _SkyAffixer_currentPlacement = new WeakMap(), _SkyAffixer_offsetChange = new WeakMap(), _SkyAffixer_offsetChangeObs = new WeakMap(), _SkyAffixer_overflowParents = new WeakMap(), _SkyAffixer_overflowScroll = new WeakMap(), _SkyAffixer_overflowScrollObs = new WeakMap(), _SkyAffixer_placementChange = new WeakMap(), _SkyAffixer_placementChangeObs = new WeakMap(), _SkyAffixer_renderer = new WeakMap(), _SkyAffixer_resizeListener = new WeakMap(), _SkyAffixer_scrollListeners = new WeakMap(), _SkyAffixer__config = new WeakMap(), _SkyAffixer_instances = new WeakSet(), _SkyAffixer_config_get = function _SkyAffixer_config_get() {
|
100
|
+
return __classPrivateFieldGet(this, _SkyAffixer__config, "f");
|
101
|
+
}, _SkyAffixer_config_set = function _SkyAffixer_config_set(value) {
|
102
|
+
const merged = {
|
103
|
+
...DEFAULT_AFFIX_CONFIG,
|
104
|
+
...value,
|
105
|
+
};
|
106
|
+
// Make sure none of the values are undefined.
|
107
|
+
let key;
|
108
|
+
for (key in merged) {
|
109
|
+
if (merged[key] === undefined) {
|
110
|
+
merged[key] = DEFAULT_AFFIX_CONFIG[key];
|
99
111
|
}
|
100
112
|
}
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
} while (!isAffixedElementFullyVisible && attempts < maxAttempts);
|
122
|
-
if (isAffixedElementFullyVisible) {
|
123
|
-
if (this.isBaseElementVisible()) {
|
124
|
-
this.notifyPlacementChange(placement);
|
125
|
-
}
|
126
|
-
else {
|
127
|
-
this.notifyPlacementChange(null);
|
128
|
-
}
|
129
|
-
return offset;
|
113
|
+
__classPrivateFieldSet(this, _SkyAffixer__config, merged, "f");
|
114
|
+
}, _SkyAffixer_affix = function _SkyAffixer_affix() {
|
115
|
+
const offset = __classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_getOffset).call(this);
|
116
|
+
if (__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_isNewOffset).call(this, offset)) {
|
117
|
+
__classPrivateFieldGet(this, _SkyAffixer_renderer, "f").setStyle(__classPrivateFieldGet(this, _SkyAffixer_affixedElement, "f"), 'top', `${offset.top}px`);
|
118
|
+
__classPrivateFieldGet(this, _SkyAffixer_renderer, "f").setStyle(__classPrivateFieldGet(this, _SkyAffixer_affixedElement, "f"), 'left', `${offset.left}px`);
|
119
|
+
__classPrivateFieldGet(this, _SkyAffixer_offsetChange, "f").next({ offset });
|
120
|
+
}
|
121
|
+
}, _SkyAffixer_getOffset = function _SkyAffixer_getOffset() {
|
122
|
+
const parent = __classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_getAutoFitContextParent).call(this);
|
123
|
+
const maxAttempts = 4;
|
124
|
+
let attempts = 0;
|
125
|
+
let isAffixedElementFullyVisible = false;
|
126
|
+
let offset;
|
127
|
+
let placement = __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).placement;
|
128
|
+
do {
|
129
|
+
offset = __classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_getPreferredOffset).call(this, placement);
|
130
|
+
isAffixedElementFullyVisible = isOffsetFullyVisibleWithinParent(parent, offset, __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).autoFitOverflowOffset);
|
131
|
+
if (!__classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).enableAutoFit) {
|
132
|
+
break;
|
130
133
|
}
|
131
|
-
if (
|
132
|
-
|
134
|
+
if (!isAffixedElementFullyVisible) {
|
135
|
+
placement =
|
136
|
+
attempts % 2 === 0
|
137
|
+
? getInversePlacement(placement)
|
138
|
+
: getNextPlacement(placement);
|
133
139
|
}
|
134
|
-
|
135
|
-
|
140
|
+
attempts++;
|
141
|
+
} while (!isAffixedElementFullyVisible && attempts < maxAttempts);
|
142
|
+
if (isAffixedElementFullyVisible) {
|
143
|
+
if (__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_isBaseElementVisible).call(this)) {
|
144
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_notifyPlacementChange).call(this, placement);
|
145
|
+
}
|
146
|
+
else {
|
147
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_notifyPlacementChange).call(this, null);
|
148
|
+
}
|
149
|
+
return offset;
|
136
150
|
}
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
}
|
160
|
-
else {
|
161
|
-
top = baseRect.bottom;
|
162
|
-
switch (verticalAlignment) {
|
163
|
-
case 'top':
|
164
|
-
default:
|
165
|
-
break;
|
166
|
-
case 'middle':
|
167
|
-
top = top - affixedRect.height / 2;
|
168
|
-
break;
|
169
|
-
case 'bottom':
|
170
|
-
top = top - affixedRect.height;
|
171
|
-
break;
|
172
|
-
}
|
173
|
-
}
|
174
|
-
switch (horizontalAlignment) {
|
175
|
-
case 'left':
|
176
|
-
left = baseRect.left;
|
151
|
+
if (__classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).enableAutoFit) {
|
152
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_notifyPlacementChange).call(this, null);
|
153
|
+
}
|
154
|
+
// No suitable placement was found, so revert to preferred placement.
|
155
|
+
return __classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_getPreferredOffset).call(this, __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).placement);
|
156
|
+
}, _SkyAffixer_getPreferredOffset = function _SkyAffixer_getPreferredOffset(placement) {
|
157
|
+
if (!__classPrivateFieldGet(this, _SkyAffixer_baseElement, "f")) {
|
158
|
+
return { top: 0, left: 0, bottom: 0, right: 0 };
|
159
|
+
}
|
160
|
+
const affixedRect = __classPrivateFieldGet(this, _SkyAffixer_affixedElement, "f").getBoundingClientRect();
|
161
|
+
const baseRect = __classPrivateFieldGet(this, _SkyAffixer_baseElement, "f").getBoundingClientRect();
|
162
|
+
const horizontalAlignment = __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).horizontalAlignment;
|
163
|
+
const verticalAlignment = __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).verticalAlignment;
|
164
|
+
const enableAutoFit = __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).enableAutoFit;
|
165
|
+
let top;
|
166
|
+
let left;
|
167
|
+
if (placement === 'above' || placement === 'below') {
|
168
|
+
if (placement === 'above') {
|
169
|
+
top = baseRect.top - affixedRect.height;
|
170
|
+
switch (verticalAlignment) {
|
171
|
+
case 'top':
|
172
|
+
top = top + affixedRect.height;
|
177
173
|
break;
|
178
|
-
case '
|
179
|
-
|
180
|
-
left = baseRect.left + baseRect.width / 2 - affixedRect.width / 2;
|
174
|
+
case 'middle':
|
175
|
+
top = top + affixedRect.height / 2;
|
181
176
|
break;
|
182
|
-
case '
|
183
|
-
|
177
|
+
case 'bottom':
|
178
|
+
default:
|
184
179
|
break;
|
185
180
|
}
|
186
181
|
}
|
187
182
|
else {
|
188
|
-
|
189
|
-
left = baseRect.left - affixedRect.width;
|
190
|
-
}
|
191
|
-
else {
|
192
|
-
left = baseRect.right;
|
193
|
-
}
|
183
|
+
top = baseRect.bottom;
|
194
184
|
switch (verticalAlignment) {
|
195
185
|
case 'top':
|
196
|
-
|
186
|
+
default:
|
197
187
|
break;
|
198
188
|
case 'middle':
|
199
|
-
|
200
|
-
top = baseRect.top + baseRect.height / 2 - affixedRect.height / 2;
|
189
|
+
top = top - affixedRect.height / 2;
|
201
190
|
break;
|
202
191
|
case 'bottom':
|
203
|
-
top =
|
192
|
+
top = top - affixedRect.height;
|
204
193
|
break;
|
205
194
|
}
|
206
195
|
}
|
207
|
-
|
208
|
-
if (enableAutoFit) {
|
209
|
-
offset = this.adjustOffsetToOverflowParent({ ...offset }, placement);
|
210
|
-
}
|
211
|
-
offset.bottom = offset.top + affixedRect.height;
|
212
|
-
offset.right = offset.left + affixedRect.width;
|
213
|
-
return offset;
|
214
|
-
}
|
215
|
-
/**
|
216
|
-
* Slightly adjust the offset to fit within the scroll parent's boundaries if
|
217
|
-
* the affixed element would otherwise be clipped.
|
218
|
-
*/
|
219
|
-
adjustOffsetToOverflowParent(offset, placement) {
|
220
|
-
const parent = this.getAutoFitContextParent();
|
221
|
-
const parentOffset = getElementOffset(parent, this.config.autoFitOverflowOffset);
|
222
|
-
const affixedRect = this.affixedRect;
|
223
|
-
const baseRect = this.baseRect;
|
224
|
-
// A pixel value representing the leeway between the edge of the overflow parent and the edge
|
225
|
-
// of the base element before it dissapears from view.
|
226
|
-
// If the visible portion of the base element is less than this pixel value, the auto-fit
|
227
|
-
// functionality attempts to find another placement.
|
228
|
-
const defaultPixelTolerance = 40;
|
229
|
-
let pixelTolerance;
|
230
|
-
const originalOffsetTop = offset.top;
|
231
|
-
const originalOffsetLeft = offset.left;
|
232
|
-
switch (placement) {
|
233
|
-
case 'above':
|
234
|
-
case 'below':
|
235
|
-
// Keep the affixed element within the overflow parent.
|
236
|
-
if (offset.left < parentOffset.left) {
|
237
|
-
offset.left = parentOffset.left;
|
238
|
-
}
|
239
|
-
else if (offset.left + affixedRect.width > parentOffset.right) {
|
240
|
-
offset.left = parentOffset.right - affixedRect.width;
|
241
|
-
}
|
242
|
-
// Use a smaller pixel tolerance if the base element width is less than the default.
|
243
|
-
pixelTolerance = Math.min(defaultPixelTolerance, baseRect.width);
|
244
|
-
// Make sure the affixed element never detaches from the base element.
|
245
|
-
if (offset.left + pixelTolerance > baseRect.right ||
|
246
|
-
offset.left + affixedRect.width - pixelTolerance < baseRect.left) {
|
247
|
-
offset.left = originalOffsetLeft;
|
248
|
-
}
|
249
|
-
break;
|
196
|
+
switch (horizontalAlignment) {
|
250
197
|
case 'left':
|
198
|
+
left = baseRect.left;
|
199
|
+
break;
|
200
|
+
case 'center':
|
201
|
+
default:
|
202
|
+
left = baseRect.left + baseRect.width / 2 - affixedRect.width / 2;
|
203
|
+
break;
|
251
204
|
case 'right':
|
252
|
-
|
253
|
-
if (offset.top < parentOffset.top) {
|
254
|
-
offset.top = parentOffset.top;
|
255
|
-
}
|
256
|
-
else if (offset.top + affixedRect.height > parentOffset.bottom) {
|
257
|
-
offset.top = parentOffset.bottom - affixedRect.height;
|
258
|
-
}
|
259
|
-
// Use a smaller pixel tolerance if the base element height is less than the default.
|
260
|
-
pixelTolerance = Math.min(defaultPixelTolerance, baseRect.height);
|
261
|
-
// Make sure the affixed element never detaches from the base element.
|
262
|
-
if (offset.top + pixelTolerance > baseRect.bottom ||
|
263
|
-
offset.top + affixedRect.height - pixelTolerance < baseRect.top) {
|
264
|
-
offset.top = originalOffsetTop;
|
265
|
-
}
|
205
|
+
left = baseRect.right - affixedRect.width;
|
266
206
|
break;
|
267
207
|
}
|
268
|
-
return offset;
|
269
|
-
}
|
270
|
-
getImmediateOverflowParent() {
|
271
|
-
return this.overflowParents[this.overflowParents.length - 1];
|
272
208
|
}
|
273
|
-
|
274
|
-
|
275
|
-
|
276
|
-
? this.getImmediateOverflowParent()
|
277
|
-
: bodyElement;
|
278
|
-
}
|
279
|
-
notifyPlacementChange(placement) {
|
280
|
-
if (this.currentPlacement !== placement) {
|
281
|
-
this.currentPlacement = placement;
|
282
|
-
this._placementChange.next({
|
283
|
-
placement,
|
284
|
-
});
|
209
|
+
else {
|
210
|
+
if (placement === 'left') {
|
211
|
+
left = baseRect.left - affixedRect.width;
|
285
212
|
}
|
286
|
-
|
287
|
-
|
288
|
-
this.removeScrollListeners();
|
289
|
-
this.removeResizeListener();
|
290
|
-
this._config =
|
291
|
-
this.affixedRect =
|
292
|
-
this.baseElement =
|
293
|
-
this.baseRect =
|
294
|
-
this.currentPlacement =
|
295
|
-
this.currentOffset =
|
296
|
-
this.overflowParents =
|
297
|
-
undefined;
|
298
|
-
}
|
299
|
-
isNewOffset(offset) {
|
300
|
-
if (this.currentOffset === undefined) {
|
301
|
-
this.currentOffset = offset;
|
302
|
-
return true;
|
213
|
+
else {
|
214
|
+
left = baseRect.right;
|
303
215
|
}
|
304
|
-
|
305
|
-
|
306
|
-
|
216
|
+
switch (verticalAlignment) {
|
217
|
+
case 'top':
|
218
|
+
top = baseRect.top;
|
219
|
+
break;
|
220
|
+
case 'middle':
|
221
|
+
default:
|
222
|
+
top = baseRect.top + baseRect.height / 2 - affixedRect.height / 2;
|
223
|
+
break;
|
224
|
+
case 'bottom':
|
225
|
+
top = baseRect.bottom - affixedRect.height;
|
226
|
+
break;
|
307
227
|
}
|
308
|
-
this.currentOffset = offset;
|
309
|
-
return true;
|
310
228
|
}
|
311
|
-
|
312
|
-
|
313
|
-
|
314
|
-
|
315
|
-
|
316
|
-
bottom: this.baseRect.bottom,
|
317
|
-
}, this.config.autoFitOverflowOffset);
|
229
|
+
const offset = { top, left, bottom: 0, right: 0 };
|
230
|
+
if (enableAutoFit) {
|
231
|
+
const adjustments = __classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_adjustOffsetToOverflowParent).call(this, { top, left }, placement, __classPrivateFieldGet(this, _SkyAffixer_baseElement, "f"));
|
232
|
+
offset.top = adjustments.top;
|
233
|
+
offset.left = adjustments.left;
|
318
234
|
}
|
319
|
-
|
320
|
-
|
321
|
-
|
322
|
-
|
323
|
-
|
324
|
-
|
325
|
-
|
235
|
+
offset.bottom = offset.top + affixedRect.height;
|
236
|
+
offset.right = offset.left + affixedRect.width;
|
237
|
+
return offset;
|
238
|
+
}, _SkyAffixer_adjustOffsetToOverflowParent = function _SkyAffixer_adjustOffsetToOverflowParent(offset, placement, baseElement) {
|
239
|
+
const parent = __classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_getAutoFitContextParent).call(this);
|
240
|
+
const parentOffset = getElementOffset(parent, __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).autoFitOverflowOffset);
|
241
|
+
const affixedRect = __classPrivateFieldGet(this, _SkyAffixer_affixedElement, "f").getBoundingClientRect();
|
242
|
+
const baseRect = baseElement.getBoundingClientRect();
|
243
|
+
// A pixel value representing the leeway between the edge of the overflow parent and the edge
|
244
|
+
// of the base element before it dissapears from view.
|
245
|
+
// If the visible portion of the base element is less than this pixel value, the auto-fit
|
246
|
+
// functionality attempts to find another placement.
|
247
|
+
const defaultPixelTolerance = 40;
|
248
|
+
let pixelTolerance;
|
249
|
+
const originalOffsetTop = offset.top;
|
250
|
+
const originalOffsetLeft = offset.left;
|
251
|
+
switch (placement) {
|
252
|
+
case 'above':
|
253
|
+
case 'below':
|
254
|
+
// Keep the affixed element within the overflow parent.
|
255
|
+
if (offset.left < parentOffset.left) {
|
256
|
+
offset.left = parentOffset.left;
|
257
|
+
}
|
258
|
+
else if (offset.left + affixedRect.width > parentOffset.right) {
|
259
|
+
offset.left = parentOffset.right - affixedRect.width;
|
260
|
+
}
|
261
|
+
// Use a smaller pixel tolerance if the base element width is less than the default.
|
262
|
+
pixelTolerance = Math.min(defaultPixelTolerance, baseRect.width);
|
263
|
+
// Make sure the affixed element never detaches from the base element.
|
264
|
+
if (offset.left + pixelTolerance > baseRect.right ||
|
265
|
+
offset.left + affixedRect.width - pixelTolerance < baseRect.left) {
|
266
|
+
offset.left = originalOffsetLeft;
|
267
|
+
}
|
268
|
+
break;
|
269
|
+
case 'left':
|
270
|
+
case 'right':
|
271
|
+
// Keep the affixed element within the overflow parent.
|
272
|
+
if (offset.top < parentOffset.top) {
|
273
|
+
offset.top = parentOffset.top;
|
274
|
+
}
|
275
|
+
else if (offset.top + affixedRect.height > parentOffset.bottom) {
|
276
|
+
offset.top = parentOffset.bottom - affixedRect.height;
|
277
|
+
}
|
278
|
+
// Use a smaller pixel tolerance if the base element height is less than the default.
|
279
|
+
pixelTolerance = Math.min(defaultPixelTolerance, baseRect.height);
|
280
|
+
// Make sure the affixed element never detaches from the base element.
|
281
|
+
if (offset.top + pixelTolerance > baseRect.bottom ||
|
282
|
+
offset.top + affixedRect.height - pixelTolerance < baseRect.top) {
|
283
|
+
offset.top = originalOffsetTop;
|
284
|
+
}
|
285
|
+
break;
|
286
|
+
}
|
287
|
+
return offset;
|
288
|
+
}, _SkyAffixer_getImmediateOverflowParent = function _SkyAffixer_getImmediateOverflowParent() {
|
289
|
+
return __classPrivateFieldGet(this, _SkyAffixer_overflowParents, "f")[__classPrivateFieldGet(this, _SkyAffixer_overflowParents, "f").length - 1];
|
290
|
+
}, _SkyAffixer_getAutoFitContextParent = function _SkyAffixer_getAutoFitContextParent() {
|
291
|
+
const bodyElement = __classPrivateFieldGet(this, _SkyAffixer_overflowParents, "f")[0];
|
292
|
+
return __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).autoFitContext === SkyAffixAutoFitContext.OverflowParent
|
293
|
+
? __classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_getImmediateOverflowParent).call(this)
|
294
|
+
: bodyElement;
|
295
|
+
}, _SkyAffixer_notifyPlacementChange = function _SkyAffixer_notifyPlacementChange(placement) {
|
296
|
+
if (__classPrivateFieldGet(this, _SkyAffixer_currentPlacement, "f") !== placement) {
|
297
|
+
__classPrivateFieldSet(this, _SkyAffixer_currentPlacement, placement ?? undefined, "f");
|
298
|
+
__classPrivateFieldGet(this, _SkyAffixer_placementChange, "f").next({
|
299
|
+
placement,
|
326
300
|
});
|
327
301
|
}
|
328
|
-
|
329
|
-
|
302
|
+
}, _SkyAffixer_reset = function _SkyAffixer_reset() {
|
303
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_removeScrollListeners).call(this);
|
304
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_removeResizeListener).call(this);
|
305
|
+
__classPrivateFieldSet(this, _SkyAffixer_overflowParents, [], "f");
|
306
|
+
__classPrivateFieldSet(this, _SkyAffixer_instances, __classPrivateFieldSet(this, _SkyAffixer_baseElement, __classPrivateFieldSet(this, _SkyAffixer_currentPlacement, __classPrivateFieldSet(this, _SkyAffixer_currentOffset, undefined, "f"), "f"), "f"), "a", _SkyAffixer_config_set);
|
307
|
+
}, _SkyAffixer_isNewOffset = function _SkyAffixer_isNewOffset(offset) {
|
308
|
+
if (__classPrivateFieldGet(this, _SkyAffixer_currentOffset, "f") === undefined) {
|
309
|
+
__classPrivateFieldSet(this, _SkyAffixer_currentOffset, offset, "f");
|
310
|
+
return true;
|
330
311
|
}
|
331
|
-
|
332
|
-
|
333
|
-
|
334
|
-
this.resizeListener = undefined;
|
335
|
-
}
|
312
|
+
if (__classPrivateFieldGet(this, _SkyAffixer_currentOffset, "f").top === offset.top &&
|
313
|
+
__classPrivateFieldGet(this, _SkyAffixer_currentOffset, "f").left === offset.left) {
|
314
|
+
return false;
|
336
315
|
}
|
337
|
-
|
338
|
-
|
339
|
-
|
340
|
-
|
341
|
-
|
342
|
-
this.scrollListeners = undefined;
|
343
|
-
}
|
316
|
+
__classPrivateFieldSet(this, _SkyAffixer_currentOffset, offset, "f");
|
317
|
+
return true;
|
318
|
+
}, _SkyAffixer_isBaseElementVisible = function _SkyAffixer_isBaseElementVisible() {
|
319
|
+
if (!__classPrivateFieldGet(this, _SkyAffixer_baseElement, "f")) {
|
320
|
+
return false;
|
344
321
|
}
|
345
|
-
|
346
|
-
//# 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 { getInversePlacement, getNextPlacement } from './affix-utils';\nimport {\n  getElementOffset,\n  getOverflowParents,\n  isOffsetFullyVisibleWithinParent,\n  isOffsetPartiallyVisibleWithinParent,\n} from './dom-utils';\n\nconst DEFAULT_AFFIX_CONFIG: SkyAffixConfig = {\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._offsetChange.asObservable();\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._overflowScroll.asObservable();\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._placementChange.asObservable();\n  }\n\n  private get config(): SkyAffixConfig {\n    return this._config;\n  }\n\n  private set config(value: SkyAffixConfig) {\n    const merged = { ...DEFAULT_AFFIX_CONFIG, ...value };\n\n    // Make sure none of the values are undefined.\n    Object.keys(merged).forEach((k: keyof SkyAffixConfig) => {\n      if (merged[k] === undefined) {\n        (merged as any)[k] = DEFAULT_AFFIX_CONFIG[k];\n      }\n    });\n\n    this._config = merged;\n  }\n\n  private affixedRect: ClientRect;\n\n  private baseElement: HTMLElement;\n\n  private baseRect: ClientRect;\n\n  private currentOffset: SkyAffixOffset;\n\n  private currentPlacement: SkyAffixPlacement;\n\n  private overflowParents: HTMLElement[];\n\n  private resizeListener: Subscription;\n\n  private scrollListeners: (() => void)[];\n\n  private _config: SkyAffixConfig;\n\n  private _offsetChange = new Subject<SkyAffixOffsetChange>();\n\n  private _overflowScroll = new Subject<void>();\n\n  private _placementChange = new Subject<SkyAffixPlacementChange>();\n\n  constructor(\n    private affixedElement: HTMLElement,\n    private renderer: Renderer2\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  /**\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    this._offsetChange =\n      this._placementChange =\n      this._overflowScroll =\n        undefined;\n  }\n\n  private affix(): void {\n    this.baseRect = this.baseElement.getBoundingClientRect();\n    this.affixedRect = this.affixedElement.getBoundingClientRect();\n\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      this._offsetChange.next({ offset });\n    }\n  }\n\n  private getOffset(): SkyAffixOffset {\n    const parent = this.getAutoFitContextParent();\n\n    const maxAttempts = 4;\n    let attempts = 0;\n\n    let isAffixedElementFullyVisible = false;\n    let offset: SkyAffixOffset;\n    let placement = this.config.placement;\n\n    do {\n      offset = this.getPreferredOffset(placement);\n      isAffixedElementFullyVisible = isOffsetFullyVisibleWithinParent(\n        parent,\n        offset,\n        this.config.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  private getPreferredOffset(placement: SkyAffixPlacement): SkyAffixOffset {\n    const affixedRect = this.affixedRect;\n    const baseRect = this.baseRect;\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    let offset: SkyAffixOffset = { left, top };\n    if (enableAutoFit) {\n      offset = this.adjustOffsetToOverflowParent({ ...offset }, placement);\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  private adjustOffsetToOverflowParent(\n    offset: SkyAffixOffset,\n    placement: SkyAffixPlacement\n  ): SkyAffixOffset {\n    const parent = this.getAutoFitContextParent();\n    const parentOffset = getElementOffset(\n      parent,\n      this.config.autoFitOverflowOffset\n    );\n\n    const affixedRect = this.affixedRect;\n    const baseRect = this.baseRect;\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 dissapears 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  private getImmediateOverflowParent(): HTMLElement {\n    return this.overflowParents[this.overflowParents.length - 1];\n  }\n\n  private getAutoFitContextParent(): HTMLElement {\n    const bodyElement = this.overflowParents[0];\n\n    return this.config.autoFitContext === SkyAffixAutoFitContext.OverflowParent\n      ? this.getImmediateOverflowParent()\n      : bodyElement;\n  }\n\n  private notifyPlacementChange(placement: SkyAffixPlacement | null): void {\n    if (this.currentPlacement !== placement) {\n      this.currentPlacement = placement;\n      this._placementChange.next({\n        placement,\n      });\n    }\n  }\n\n  private reset(): void {\n    this.removeScrollListeners();\n    this.removeResizeListener();\n\n    this._config =\n      this.affixedRect =\n      this.baseElement =\n      this.baseRect =\n      this.currentPlacement =\n      this.currentOffset =\n      this.overflowParents =\n        undefined;\n  }\n\n  private 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  private isBaseElementVisible(): boolean {\n    return isOffsetPartiallyVisibleWithinParent(\n      this.getImmediateOverflowParent(),\n      {\n        top: this.baseRect.top,\n        left: this.baseRect.left,\n        right: this.baseRect.right,\n        bottom: this.baseRect.bottom,\n      },\n      this.config.autoFitOverflowOffset\n    );\n  }\n\n  private 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  private addResizeListener(): void {\n    this.resizeListener = fromEvent(window, 'resize').subscribe(() =>\n      this.affix()\n    );\n  }\n\n  private removeResizeListener(): void {\n    if (this.resizeListener) {\n      this.resizeListener.unsubscribe();\n      this.resizeListener = undefined;\n    }\n  }\n\n  private 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"]}
|
322
|
+
const baseRect = __classPrivateFieldGet(this, _SkyAffixer_baseElement, "f").getBoundingClientRect();
|
323
|
+
return isOffsetPartiallyVisibleWithinParent(__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_getImmediateOverflowParent).call(this), {
|
324
|
+
top: baseRect.top,
|
325
|
+
left: baseRect.left,
|
326
|
+
right: baseRect.right,
|
327
|
+
bottom: baseRect.bottom,
|
328
|
+
}, __classPrivateFieldGet(this, _SkyAffixer_instances, "a", _SkyAffixer_config_get).autoFitOverflowOffset);
|
329
|
+
}, _SkyAffixer_addScrollListeners = function _SkyAffixer_addScrollListeners() {
|
330
|
+
__classPrivateFieldSet(this, _SkyAffixer_scrollListeners, __classPrivateFieldGet(this, _SkyAffixer_overflowParents, "f").map((parentElement) => {
|
331
|
+
const overflow = parentElement === document.body ? 'window' : parentElement;
|
332
|
+
return __classPrivateFieldGet(this, _SkyAffixer_renderer, "f").listen(overflow, 'scroll', () => {
|
333
|
+
__classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_affix).call(this);
|
334
|
+
__classPrivateFieldGet(this, _SkyAffixer_overflowScroll, "f").next();
|
335
|
+
});
|
336
|
+
}), "f");
|
337
|
+
}, _SkyAffixer_addResizeListener = function _SkyAffixer_addResizeListener() {
|
338
|
+
__classPrivateFieldSet(this, _SkyAffixer_resizeListener, fromEvent(window, 'resize').subscribe(() => __classPrivateFieldGet(this, _SkyAffixer_instances, "m", _SkyAffixer_affix).call(this)), "f");
|
339
|
+
}, _SkyAffixer_removeResizeListener = function _SkyAffixer_removeResizeListener() {
|
340
|
+
if (__classPrivateFieldGet(this, _SkyAffixer_resizeListener, "f")) {
|
341
|
+
__classPrivateFieldGet(this, _SkyAffixer_resizeListener, "f").unsubscribe();
|
342
|
+
__classPrivateFieldSet(this, _SkyAffixer_resizeListener, undefined, "f");
|
343
|
+
}
|
344
|
+
}, _SkyAffixer_removeScrollListeners = function _SkyAffixer_removeScrollListeners() {
|
345
|
+
if (__classPrivateFieldGet(this, _SkyAffixer_scrollListeners, "f")) {
|
346
|
+
// Remove renderer-generated listeners by calling the listener itself.
|
347
|
+
// https://github.com/angular/angular/issues/9368#issuecomment-227199778
|
348
|
+
__classPrivateFieldGet(this, _SkyAffixer_scrollListeners, "f").forEach((listener) => listener());
|
349
|
+
__classPrivateFieldSet(this, _SkyAffixer_scrollListeners, undefined, "f");
|
350
|
+
}
|
351
|
+
};
|
352
|
+
//# 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 { 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    do {\n      offset = this.#getPreferredOffset(placement);\n      isAffixedElementFullyVisible = isOffsetFullyVisibleWithinParent(\n        parent,\n        offset,\n        this.#config.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  #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.#affixedElement.getBoundingClientRect();\n    const baseRect = this.#baseElement.getBoundingClientRect();\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.#affixedElement.getBoundingClientRect();\n    const baseRect = baseElement.getBoundingClientRect();\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 dissapears 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"]}
|