@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.
- package/documentation.json +29 -29
- package/{esm2020 → esm2022}/lib/modules/adapter-service/adapter.module.mjs +5 -5
- package/esm2022/lib/modules/adapter-service/adapter.service.mjs +225 -0
- package/esm2022/lib/modules/affix/affix.directive.mjs +116 -0
- package/{esm2020 → esm2022}/lib/modules/affix/affix.module.mjs +5 -5
- package/esm2022/lib/modules/affix/affix.service.mjs +25 -0
- package/esm2022/lib/modules/affix/affixer.mjs +398 -0
- package/esm2022/lib/modules/default-input-provider/default-input-provider.mjs +26 -0
- package/esm2022/lib/modules/dock/dock-dom-adapter.service.mjs +81 -0
- package/esm2022/lib/modules/dock/dock-item.mjs +31 -0
- package/esm2022/lib/modules/dock/dock.component.mjs +108 -0
- package/{esm2020 → esm2022}/lib/modules/dock/dock.module.mjs +5 -5
- package/esm2022/lib/modules/dock/dock.service.mjs +96 -0
- package/{esm2020 → esm2022}/lib/modules/dynamic-component/dynamic-component.module.mjs +5 -5
- package/esm2022/lib/modules/dynamic-component/dynamic-component.service.mjs +106 -0
- package/{esm2020 → esm2022}/lib/modules/format/app-format.mjs +4 -4
- package/esm2022/lib/modules/id/id.directive.mjs +31 -0
- package/{esm2020 → esm2022}/lib/modules/id/id.module.mjs +5 -5
- package/{esm2020 → esm2022}/lib/modules/id/id.service.mjs +4 -4
- package/esm2022/lib/modules/layout-host/layout-host.service.mjs +22 -0
- package/esm2022/lib/modules/live-announcer/live-announcer.service.mjs +79 -0
- package/{esm2020 → esm2022}/lib/modules/log/log.module.mjs +5 -5
- package/esm2022/lib/modules/log/log.service.mjs +148 -0
- package/{esm2020 → esm2022}/lib/modules/media-query/media-query.module.mjs +5 -5
- package/esm2022/lib/modules/media-query/media-query.service.mjs +123 -0
- package/{esm2020 → esm2022}/lib/modules/mutation/mutation-observer-service.mjs +4 -4
- package/{esm2020 → esm2022}/lib/modules/numeric/numeric.module.mjs +5 -5
- package/esm2022/lib/modules/numeric/numeric.pipe.mjs +81 -0
- package/esm2022/lib/modules/numeric/numeric.service.mjs +205 -0
- package/esm2022/lib/modules/overlay/overlay-adapter.service.mjs +44 -0
- package/esm2022/lib/modules/overlay/overlay-instance.mjs +62 -0
- package/esm2022/lib/modules/overlay/overlay.component.mjs +193 -0
- package/{esm2020 → esm2022}/lib/modules/overlay/overlay.module.mjs +5 -5
- package/esm2022/lib/modules/overlay/overlay.service.mjs +124 -0
- package/{esm2020 → esm2022}/lib/modules/percent-pipe/percent-pipe.module.mjs +5 -5
- package/esm2022/lib/modules/percent-pipe/percent.pipe.mjs +56 -0
- package/esm2022/lib/modules/resize-observer/resize-observer-media-query.service.mjs +120 -0
- package/esm2022/lib/modules/resize-observer/resize-observer.service.mjs +77 -0
- package/esm2022/lib/modules/scrollable-host/scrollable-host.service.mjs +223 -0
- package/esm2022/lib/modules/shared/number-format/number-format-utility.mjs +72 -0
- package/esm2022/lib/modules/shared/sky-core-resources.module.mjs +50 -0
- package/esm2022/lib/modules/title/title.service.mjs +31 -0
- package/esm2022/lib/modules/trim/trim.directive.mjs +64 -0
- package/{esm2020 → esm2022}/lib/modules/trim/trim.module.mjs +5 -5
- package/{esm2020 → esm2022}/lib/modules/ui-config/ui-config.service.mjs +4 -4
- package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper-host-options.mjs +4 -4
- package/esm2022/lib/modules/viewkeeper/viewkeeper.directive.mjs +131 -0
- package/esm2022/lib/modules/viewkeeper/viewkeeper.mjs +283 -0
- package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper.module.mjs +5 -5
- package/esm2022/lib/modules/viewkeeper/viewkeeper.service.mjs +39 -0
- package/{esm2020 → esm2022}/lib/modules/window/window-ref.mjs +4 -4
- package/esm2022/testing/core-testing.module.mjs +32 -0
- package/esm2022/testing/mock-media-query.service.mjs +46 -0
- package/{esm2020 → esm2022}/testing/mock-ui-config.service.mjs +4 -4
- package/esm2022/testing/overlay/overlay-harness.mjs +43 -0
- package/esm2022/testing/shared/component-harness.mjs +13 -0
- package/{fesm2020 → fesm2022}/skyux-core-testing.mjs +28 -34
- package/{fesm2020 → fesm2022}/skyux-core-testing.mjs.map +1 -1
- package/fesm2022/skyux-core.mjs +3858 -0
- package/fesm2022/skyux-core.mjs.map +1 -0
- package/lib/modules/affix/affix.directive.d.ts +1 -1
- package/lib/modules/viewkeeper/viewkeeper.directive.d.ts +1 -1
- package/package.json +14 -22
- package/esm2020/lib/modules/adapter-service/adapter.service.mjs +0 -227
- package/esm2020/lib/modules/affix/affix.directive.mjs +0 -118
- package/esm2020/lib/modules/affix/affix.service.mjs +0 -28
- package/esm2020/lib/modules/affix/affixer.mjs +0 -377
- package/esm2020/lib/modules/default-input-provider/default-input-provider.mjs +0 -31
- package/esm2020/lib/modules/dock/dock-dom-adapter.service.mjs +0 -83
- package/esm2020/lib/modules/dock/dock-item.mjs +0 -34
- package/esm2020/lib/modules/dock/dock.component.mjs +0 -110
- package/esm2020/lib/modules/dock/dock.service.mjs +0 -98
- package/esm2020/lib/modules/dynamic-component/dynamic-component.service.mjs +0 -109
- package/esm2020/lib/modules/id/id.directive.mjs +0 -34
- package/esm2020/lib/modules/layout-host/layout-host.service.mjs +0 -27
- package/esm2020/lib/modules/live-announcer/live-announcer.service.mjs +0 -80
- package/esm2020/lib/modules/log/log.service.mjs +0 -147
- package/esm2020/lib/modules/media-query/media-query.service.mjs +0 -124
- package/esm2020/lib/modules/numeric/numeric.pipe.mjs +0 -84
- package/esm2020/lib/modules/numeric/numeric.service.mjs +0 -179
- package/esm2020/lib/modules/overlay/overlay-adapter.service.mjs +0 -47
- package/esm2020/lib/modules/overlay/overlay-instance.mjs +0 -65
- package/esm2020/lib/modules/overlay/overlay.component.mjs +0 -192
- package/esm2020/lib/modules/overlay/overlay.service.mjs +0 -125
- package/esm2020/lib/modules/percent-pipe/percent.pipe.mjs +0 -59
- package/esm2020/lib/modules/resize-observer/resize-observer-media-query.service.mjs +0 -121
- package/esm2020/lib/modules/resize-observer/resize-observer.service.mjs +0 -79
- package/esm2020/lib/modules/scrollable-host/scrollable-host.service.mjs +0 -218
- package/esm2020/lib/modules/shared/number-format/number-format-utility.mjs +0 -72
- package/esm2020/lib/modules/shared/sky-core-resources.module.mjs +0 -50
- package/esm2020/lib/modules/title/title.service.mjs +0 -34
- package/esm2020/lib/modules/trim/trim.directive.mjs +0 -65
- package/esm2020/lib/modules/viewkeeper/viewkeeper.directive.mjs +0 -131
- package/esm2020/lib/modules/viewkeeper/viewkeeper.mjs +0 -276
- package/esm2020/lib/modules/viewkeeper/viewkeeper.service.mjs +0 -42
- package/esm2020/testing/core-testing.module.mjs +0 -32
- package/esm2020/testing/mock-media-query.service.mjs +0 -46
- package/esm2020/testing/overlay/overlay-harness.mjs +0 -43
- package/esm2020/testing/shared/component-harness.mjs +0 -19
- package/fesm2015/skyux-core-testing.mjs +0 -209
- package/fesm2015/skyux-core-testing.mjs.map +0 -1
- package/fesm2015/skyux-core.mjs +0 -3835
- package/fesm2015/skyux-core.mjs.map +0 -1
- package/fesm2020/skyux-core.mjs +0 -3825
- package/fesm2020/skyux-core.mjs.map +0 -1
- /package/{esm2020 → esm2022}/index.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/adapter-service/focusable-children-options.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-auto-fit-context.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-config.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-horizontal-alignment.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-offset-change.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-offset.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-placement-change.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-placement.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-position.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-rect.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-utils.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/affix-vertical-alignment.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/affix/dom-utils.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dock/dock-insert-component-config.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dock/dock-item-config.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dock/dock-item-reference.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dock/dock-location.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dock/dock-options.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dock/sort-by-stack-order.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dynamic-component/dynamic-component-location.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/dynamic-component/dynamic-component-options.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/layout-host/layout-host-for-child-args.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/live-announcer/types/live-announcer-args.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/live-announcer/types/live-announcer-politeness.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/log/types/log-deprecation-args.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/log/types/log-level-token.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/log/types/log-level.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/media-query/media-breakpoints.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/media-query/media-query-listener.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/numeric/numeric-symbol.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/numeric/numeric.options.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/overlay/overlay-config.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/overlay/overlay-context.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/overlay/overlay-position.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/stacking-context/stacking-context-token.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/stacking-context/stacking-context.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/title/set-title-args.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper-boundary-info.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper-fixed-styles.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper-offset.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/viewkeeper/viewkeeper-options.mjs +0 -0
- /package/{esm2020 → esm2022}/skyux-core.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/overlay/overlay-harness-filters.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/public-api.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/shared/harness-filters.mjs +0 -0
- /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=
|