@skyux/core 5.1.3 → 5.2.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/bundles/skyux-core.umd.js +98 -14
- package/bundles/skyux-core.umd.js.map +1 -1
- package/documentation.json +132 -144
- package/esm2015/modules/scrollable-host/scrollable-host.service.js +88 -16
- package/esm2015/modules/viewkeeper/viewkeeper.directive.js +2 -2
- package/fesm2015/skyux-core.js +88 -16
- package/fesm2015/skyux-core.js.map +1 -1
- package/modules/scrollable-host/scrollable-host.service.d.ts +21 -2
- package/package.json +1 -1
@@ -1,33 +1,105 @@
|
|
1
1
|
import { Injectable } from "@angular/core";
|
2
|
-
import {
|
3
|
-
import {
|
2
|
+
import { fromEvent, Observable, Subject } from "rxjs";
|
3
|
+
import { takeUntil } from "rxjs/operators";
|
4
4
|
import * as i0 from "@angular/core";
|
5
5
|
import * as i1 from "../mutation/mutation-observer-service";
|
6
6
|
import * as i2 from "../window/window-ref";
|
7
|
+
function notifySubscribers(subscribers, item) {
|
8
|
+
for (const subscriber of subscribers) {
|
9
|
+
subscriber.next(item);
|
10
|
+
}
|
11
|
+
}
|
7
12
|
export class SkyScrollableHostService {
|
8
13
|
constructor(mutationObserverSvc, windowRef) {
|
9
14
|
this.mutationObserverSvc = mutationObserverSvc;
|
10
15
|
this.windowRef = windowRef;
|
11
16
|
}
|
12
|
-
|
17
|
+
/**
|
18
|
+
* Returns the given element's current scrollable host
|
19
|
+
* @param elementRef The element whose scrollable host is being requested
|
20
|
+
* @returns The current scrollable host
|
21
|
+
*/
|
22
|
+
getScrollableHost(elementRef) {
|
13
23
|
return this.findScrollableHost(elementRef.nativeElement);
|
14
24
|
}
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
25
|
+
/**
|
26
|
+
* Returns an observable which emits the given element's current scrollable host
|
27
|
+
* @param elementRef The element whose scrollable host is being requested
|
28
|
+
* @param completionObservable An observable which alerts the internal observers that they should complete
|
29
|
+
* @returns An observable which emits the current scrollable host
|
30
|
+
* @internal
|
31
|
+
*/
|
32
|
+
watchScrollableHost(elementRef) {
|
33
|
+
let subscribers = [];
|
34
|
+
let mutationObserver;
|
35
|
+
return new Observable((subscriber) => {
|
36
|
+
subscribers.push(subscriber);
|
37
|
+
let scrollableHost = this.findScrollableHost(elementRef.nativeElement);
|
38
|
+
if (subscribers.length === 1) {
|
39
|
+
mutationObserver = this.mutationObserverSvc.create(() => {
|
40
|
+
let newScrollableHost = this.findScrollableHost(elementRef.nativeElement);
|
41
|
+
if (newScrollableHost !== scrollableHost) {
|
42
|
+
scrollableHost = newScrollableHost;
|
43
|
+
this.observeForScrollableHostChanges(scrollableHost, mutationObserver);
|
44
|
+
notifySubscribers(subscribers, scrollableHost);
|
45
|
+
}
|
46
|
+
});
|
22
47
|
this.observeForScrollableHostChanges(scrollableHost, mutationObserver);
|
23
|
-
behaviorSubject.next(scrollableHost);
|
24
48
|
}
|
49
|
+
subscriber.next(scrollableHost);
|
50
|
+
subscriber.add(() => {
|
51
|
+
const subIndex = subscribers.indexOf(subscriber);
|
52
|
+
/* sanity check */
|
53
|
+
/* istanbul ignore else */
|
54
|
+
if (subIndex >= 0) {
|
55
|
+
subscribers.splice(subIndex, 1);
|
56
|
+
}
|
57
|
+
if (subscribers.length === 0) {
|
58
|
+
mutationObserver.disconnect();
|
59
|
+
}
|
60
|
+
});
|
25
61
|
});
|
26
|
-
|
27
|
-
|
28
|
-
|
62
|
+
}
|
63
|
+
/**
|
64
|
+
* Returns an observable which emits whenever the element's scrollable host emits a scroll event. The observable will always emit the scroll events from the elements current scrollable host and will update based on any scrollable host changes. The observable will also emit once whenever the scrollable host changes.
|
65
|
+
* @param elementRef The element whose scrollable host scroll events are being requested
|
66
|
+
* @param completionObservable An observable which alerts the internal observers that they should complete
|
67
|
+
* @returns An observable which emits the scroll events from the given element's scrollable host
|
68
|
+
*/
|
69
|
+
watchScrollableHostScrollEvents(elementRef) {
|
70
|
+
let subscribers = [];
|
71
|
+
let newScrollableHostObservable = new Subject();
|
72
|
+
let scrollableHostSubscription;
|
73
|
+
let scrollEventSubscription;
|
74
|
+
return new Observable((subscriber) => {
|
75
|
+
subscribers.push(subscriber);
|
76
|
+
if (subscribers.length === 1) {
|
77
|
+
scrollableHostSubscription = this.watchScrollableHost(elementRef)
|
78
|
+
.subscribe((scrollableHost) => {
|
79
|
+
newScrollableHostObservable.next();
|
80
|
+
newScrollableHostObservable.complete();
|
81
|
+
newScrollableHostObservable = new Subject();
|
82
|
+
scrollEventSubscription = fromEvent(scrollableHost, 'scroll')
|
83
|
+
.pipe(takeUntil(newScrollableHostObservable))
|
84
|
+
.subscribe(() => {
|
85
|
+
notifySubscribers(subscribers);
|
86
|
+
});
|
87
|
+
});
|
88
|
+
}
|
89
|
+
subscriber.add(() => {
|
90
|
+
const subIndex = subscribers.indexOf(subscriber);
|
91
|
+
/* sanity check */
|
92
|
+
/* istanbul ignore else */
|
93
|
+
if (subIndex >= 0) {
|
94
|
+
subscribers.splice(subIndex, 1);
|
95
|
+
}
|
96
|
+
if (subscribers.length === 0) {
|
97
|
+
scrollableHostSubscription.unsubscribe();
|
98
|
+
scrollEventSubscription.unsubscribe();
|
99
|
+
newScrollableHostObservable.complete();
|
100
|
+
}
|
101
|
+
});
|
29
102
|
});
|
30
|
-
return behaviorSubject;
|
31
103
|
}
|
32
104
|
findScrollableHost(element) {
|
33
105
|
const regex = /(auto|scroll)/;
|
@@ -80,4 +152,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.13", ngImpo
|
|
80
152
|
providedIn: 'root'
|
81
153
|
}]
|
82
154
|
}], ctorParameters: function () { return [{ type: i1.MutationObserverService }, { type: i2.SkyAppWindowRef }]; } });
|
83
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2Nyb2xsYWJsZS1ob3N0LnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3NyYy9tb2R1bGVzL3Njcm9sbGFibGUtaG9zdC9zY3JvbGxhYmxlLWhvc3Quc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQWMsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3ZELE9BQU8sRUFBRSxlQUFlLEVBQWMsTUFBTSxNQUFNLENBQUM7QUFDbkQsT0FBTyxFQUFFLElBQUksRUFBRSxNQUFNLGdCQUFnQixDQUFDOzs7O0FBT3RDLE1BQU0sT0FBTyx3QkFBd0I7SUFFbkMsWUFDVSxtQkFBNEMsRUFDNUMsU0FBMEI7UUFEMUIsd0JBQW1CLEdBQW5CLG1CQUFtQixDQUF5QjtRQUM1QyxjQUFTLEdBQVQsU0FBUyxDQUFpQjtJQUNoQyxDQUFDO0lBRUUsZ0JBQWdCLENBQUMsVUFBc0I7UUFDNUMsT0FBTyxJQUFJLENBQUMsa0JBQWtCLENBQUMsVUFBVSxDQUFDLGFBQWEsQ0FBQyxDQUFDO0lBQzNELENBQUM7SUFFTSxtQkFBbUIsQ0FBQyxVQUFzQixFQUFFLG9CQUFzQztRQUN2RixJQUFJLGNBQWMsR0FBRyxJQUFJLENBQUMsa0JBQWtCLENBQUMsVUFBVSxDQUFDLGFBQWEsQ0FBQyxDQUFDO1FBQ3ZFLElBQUksZUFBZSxHQUFHLElBQUksZUFBZSxDQUFDLGNBQWMsQ0FBQyxDQUFDO1FBRTFELE1BQU0sZ0JBQWdCLEdBQUcsSUFBSSxDQUFDLG1CQUFtQixDQUFDLE1BQU0sQ0FBQyxHQUFHLEVBQUU7WUFDNUQsSUFBSSxpQkFBaUIsR0FBRyxJQUFJLENBQUMsa0JBQWtCLENBQUMsVUFBVSxDQUFDLGFBQWEsQ0FBQyxDQUFDO1lBRTFFLElBQUksaUJBQWlCLEtBQUssY0FBYyxFQUFFO2dCQUN4QyxjQUFjLEdBQUcsaUJBQWlCLENBQUM7Z0JBQ25DLElBQUksQ0FBQywrQkFBK0IsQ0FBQyxjQUFjLEVBQUUsZ0JBQWdCLENBQUMsQ0FBQztnQkFDdkUsZUFBZSxDQUFDLElBQUksQ0FBQyxjQUFjLENBQUMsQ0FBQzthQUN0QztRQUNILENBQUMsQ0FBQyxDQUFDO1FBQ0gsSUFBSSxDQUFDLCtCQUErQixDQUFDLGNBQWMsRUFBRSxnQkFBZ0IsQ0FBQyxDQUFDO1FBRXZFLG9CQUFvQixDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUMsR0FBRyxFQUFFO1lBQ2hELGdCQUFnQixDQUFDLFVBQVUsRUFBRSxDQUFDO1FBQ2hDLENBQUMsQ0FBQyxDQUFBO1FBRUYsT0FBTyxlQUFlLENBQUM7SUFDekIsQ0FBQztJQUVPLGtCQUFrQixDQUFDLE9BQW9CO1FBQzdDLE1BQU0sS0FBSyxHQUFHLGVBQWUsQ0FBQztRQUM5QixNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLFlBQVksQ0FBQztRQUM5QyxNQUFNLE9BQU8sR0FBRyxTQUFTLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQztRQUV4QyxrQkFBa0I7UUFDbEIsSUFBSSxDQUFDLE9BQU8sRUFBRTtZQUNaLE9BQU8sU0FBUyxDQUFDO1NBQ2xCO1FBRUQsSUFBSSxLQUFLLEdBQUcsU0FBUyxDQUFDLGdCQUFnQixDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ2hELElBQUksTUFBTSxHQUFnQixPQUFPLENBQUM7UUFFbEMsR0FBRztZQUNELE1BQU0sR0FBaUIsTUFBTSxDQUFDLFVBQVUsQ0FBQztZQUV6QyxvR0FBb0c7WUFDcEcsSUFBSSxDQUFDLENBQUMsTUFBTSxZQUFZLFdBQVcsQ0FBQyxFQUFFO2dCQUNwQyxPQUFPLFNBQVMsQ0FBQTthQUNqQjtZQUVELEtBQUssR0FBRyxTQUFTLENBQUMsZ0JBQWdCLENBQUMsTUFBTSxDQUFDLENBQUM7U0FDNUMsUUFDQyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQztZQUMzQixDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQVMsQ0FBQztZQUM1QixNQUFNLEtBQUssT0FBTyxFQUNsQjtRQUVGLElBQUksTUFBTSxLQUFLLE9BQU8sRUFBRTtZQUN0QixPQUFPLFNBQVMsQ0FBQztTQUNsQjtRQUVELE9BQU8sTUFBTSxDQUFDO0lBQ2hCLENBQUM7SUFFTywrQkFBK0IsQ0FBQyxPQUE2QixFQUFFLGdCQUFrQztRQUN2RyxnQkFBZ0IsQ0FBQyxVQUFVLEVBQUUsQ0FBQztRQUM5QixJQUFJLE9BQU8sWUFBWSxXQUFXLEVBQUU7WUFDbEMsZ0JBQWdCLENBQUMsT0FBTyxDQUFDLE9BQU8sRUFBRTtnQkFDaEMsVUFBVSxFQUFFLElBQUk7Z0JBQ2hCLGVBQWUsRUFBRSxDQUFDLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxrQkFBa0IsQ0FBQztnQkFDaEUsT0FBTyxFQUFFLElBQUk7YUFDZCxDQUFDLENBQUM7U0FDSjthQUFNO1lBQ0wsZ0JBQWdCLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxlQUFlLEVBQUU7Z0JBQ2pELFVBQVUsRUFBRSxJQUFJO2dCQUNoQixlQUFlLEVBQUUsQ0FBQyxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsa0JBQWtCLENBQUM7Z0JBQ2hFLE9BQU8sRUFBRSxJQUFJO2FBQ2QsQ0FBQyxDQUFDO1NBQ0o7SUFDSCxDQUFDOztzSEFuRlUsd0JBQXdCOzBIQUF4Qix3QkFBd0IsY0FGdkIsTUFBTTs0RkFFUCx3QkFBd0I7a0JBSHBDLFVBQVU7bUJBQUM7b0JBQ1YsVUFBVSxFQUFFLE1BQU07aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRWxlbWVudFJlZiwgSW5qZWN0YWJsZSB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5pbXBvcnQgeyBCZWhhdmlvclN1YmplY3QsIE9ic2VydmFibGUgfSBmcm9tIFwicnhqc1wiO1xuaW1wb3J0IHsgdGFrZSB9IGZyb20gXCJyeGpzL29wZXJhdG9yc1wiO1xuaW1wb3J0IHsgTXV0YXRpb25PYnNlcnZlclNlcnZpY2UgfSBmcm9tIFwiLi4vbXV0YXRpb24vbXV0YXRpb24tb2JzZXJ2ZXItc2VydmljZVwiO1xuaW1wb3J0IHsgU2t5QXBwV2luZG93UmVmIH0gZnJvbSBcIi4uL3dpbmRvdy93aW5kb3ctcmVmXCI7XG5cbkBJbmplY3RhYmxlKHtcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnXG59KVxuZXhwb3J0IGNsYXNzIFNreVNjcm9sbGFibGVIb3N0U2VydmljZSB7XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSBtdXRhdGlvbk9ic2VydmVyU3ZjOiBNdXRhdGlvbk9ic2VydmVyU2VydmljZSxcbiAgICBwcml2YXRlIHdpbmRvd1JlZjogU2t5QXBwV2luZG93UmVmXG4gICkgeyB9XG5cbiAgcHVibGljIGdldFNjcm9sbGFiZUhvc3QoZWxlbWVudFJlZjogRWxlbWVudFJlZik6IEhUTUxFbGVtZW50IHwgV2luZG93IHtcbiAgICByZXR1cm4gdGhpcy5maW5kU2Nyb2xsYWJsZUhvc3QoZWxlbWVudFJlZi5uYXRpdmVFbGVtZW50KTtcbiAgfVxuXG4gIHB1YmxpYyB3YXRjaFNjcm9sbGFibGVIb3N0KGVsZW1lbnRSZWY6IEVsZW1lbnRSZWYsIGNvbXBsZXRpb25PYnNlcnZhYmxlOiBPYnNlcnZhYmxlPHZvaWQ+KTogT2JzZXJ2YWJsZTxIVE1MRWxlbWVudCB8IFdpbmRvdz4ge1xuICAgIGxldCBzY3JvbGxhYmxlSG9zdCA9IHRoaXMuZmluZFNjcm9sbGFibGVIb3N0KGVsZW1lbnRSZWYubmF0aXZlRWxlbWVudCk7XG4gICAgbGV0IGJlaGF2aW9yU3ViamVjdCA9IG5ldyBCZWhhdmlvclN1YmplY3Qoc2Nyb2xsYWJsZUhvc3QpO1xuXG4gICAgY29uc3QgbXV0YXRpb25PYnNlcnZlciA9IHRoaXMubXV0YXRpb25PYnNlcnZlclN2Yy5jcmVhdGUoKCkgPT4ge1xuICAgICAgbGV0IG5ld1Njcm9sbGFibGVIb3N0ID0gdGhpcy5maW5kU2Nyb2xsYWJsZUhvc3QoZWxlbWVudFJlZi5uYXRpdmVFbGVtZW50KTtcblxuICAgICAgaWYgKG5ld1Njcm9sbGFibGVIb3N0ICE9PSBzY3JvbGxhYmxlSG9zdCkge1xuICAgICAgICBzY3JvbGxhYmxlSG9zdCA9IG5ld1Njcm9sbGFibGVIb3N0O1xuICAgICAgICB0aGlzLm9ic2VydmVGb3JTY3JvbGxhYmxlSG9zdENoYW5nZXMoc2Nyb2xsYWJsZUhvc3QsIG11dGF0aW9uT2JzZXJ2ZXIpO1xuICAgICAgICBiZWhhdmlvclN1YmplY3QubmV4dChzY3JvbGxhYmxlSG9zdCk7XG4gICAgICB9XG4gICAgfSk7XG4gICAgdGhpcy5vYnNlcnZlRm9yU2Nyb2xsYWJsZUhvc3RDaGFuZ2VzKHNjcm9sbGFibGVIb3N0LCBtdXRhdGlvbk9ic2VydmVyKTtcblxuICAgIGNvbXBsZXRpb25PYnNlcnZhYmxlLnBpcGUodGFrZSgxKSkuc3Vic2NyaWJlKCgpID0+IHtcbiAgICAgIG11dGF0aW9uT2JzZXJ2ZXIuZGlzY29ubmVjdCgpO1xuICAgIH0pXG5cbiAgICByZXR1cm4gYmVoYXZpb3JTdWJqZWN0O1xuICB9XG5cbiAgcHJpdmF0ZSBmaW5kU2Nyb2xsYWJsZUhvc3QoZWxlbWVudDogSFRNTEVsZW1lbnQpOiBIVE1MRWxlbWVudCB8IFdpbmRvdyB7XG4gICAgY29uc3QgcmVnZXggPSAvKGF1dG98c2Nyb2xsKS87XG4gICAgY29uc3Qgd2luZG93T2JqID0gdGhpcy53aW5kb3dSZWYubmF0aXZlV2luZG93O1xuICAgIGNvbnN0IGJvZHlPYmogPSB3aW5kb3dPYmouZG9jdW1lbnQuYm9keTtcblxuICAgIC8qIFNhbml0eSBjaGVjayAqL1xuICAgIGlmICghZWxlbWVudCkge1xuICAgICAgcmV0dXJuIHdpbmRvd09iajtcbiAgICB9XG5cbiAgICBsZXQgc3R5bGUgPSB3aW5kb3dPYmouZ2V0Q29tcHV0ZWRTdHlsZShlbGVtZW50KTtcbiAgICBsZXQgcGFyZW50OiBIVE1MRWxlbWVudCA9IGVsZW1lbnQ7XG5cbiAgICBkbyB7XG4gICAgICBwYXJlbnQgPSA8SFRNTEVsZW1lbnQ+IHBhcmVudC5wYXJlbnROb2RlO1xuXG4gICAgICAvKiBTYW5pdHkgY2hlY2sgZm9yIGlmIHRoaXMgZnVuY3Rpb24gaXMgY2FsbGVkIGZvciBhbiBlbGVtZW50IHdoaWNoIGhhcyBiZWVuIHJlbW92ZWQgZnJvbSB0aGUgRE9NICovXG4gICAgICBpZiAoIShwYXJlbnQgaW5zdGFuY2VvZiBIVE1MRWxlbWVudCkpIHtcbiAgICAgICAgcmV0dXJuIHdpbmRvd09ialxuICAgICAgfVxuXG4gICAgICBzdHlsZSA9IHdpbmRvd09iai5nZXRDb21wdXRlZFN0eWxlKHBhcmVudCk7XG4gICAgfSB3aGlsZSAoXG4gICAgICAhcmVnZXgudGVzdChzdHlsZS5vdmVyZmxvdykgJiZcbiAgICAgICFyZWdleC50ZXN0KHN0eWxlLm92ZXJmbG93WSkgJiZcbiAgICAgIHBhcmVudCAhPT0gYm9keU9ialxuICAgICk7XG5cbiAgICBpZiAocGFyZW50ID09PSBib2R5T2JqKSB7XG4gICAgICByZXR1cm4gd2luZG93T2JqO1xuICAgIH1cblxuICAgIHJldHVybiBwYXJlbnQ7XG4gIH1cblxuICBwcml2YXRlIG9ic2VydmVGb3JTY3JvbGxhYmxlSG9zdENoYW5nZXMoZWxlbWVudDogSFRNTEVsZW1lbnQgfCBXaW5kb3csIG11dGF0aW9uT2JzZXJ2ZXI6IE11dGF0aW9uT2JzZXJ2ZXIpIHtcbiAgICBtdXRhdGlvbk9ic2VydmVyLmRpc2Nvbm5lY3QoKTtcbiAgICBpZiAoZWxlbWVudCBpbnN0YW5jZW9mIEhUTUxFbGVtZW50KSB7XG4gICAgICBtdXRhdGlvbk9ic2VydmVyLm9ic2VydmUoZWxlbWVudCwge1xuICAgICAgICBhdHRyaWJ1dGVzOiB0cnVlLFxuICAgICAgICBhdHRyaWJ1dGVGaWx0ZXI6IFtcImNsYXNzXCIsIFwic3R5bGUub3ZlcmZsb3dcIiwgXCJzdHlsZS5vdmVyZmxvdy15XCJdLFxuICAgICAgICBzdWJ0cmVlOiB0cnVlXG4gICAgICB9KTtcbiAgICB9IGVsc2Uge1xuICAgICAgbXV0YXRpb25PYnNlcnZlci5vYnNlcnZlKGRvY3VtZW50LmRvY3VtZW50RWxlbWVudCwge1xuICAgICAgICBhdHRyaWJ1dGVzOiB0cnVlLFxuICAgICAgICBhdHRyaWJ1dGVGaWx0ZXI6IFtcImNsYXNzXCIsIFwic3R5bGUub3ZlcmZsb3dcIiwgXCJzdHlsZS5vdmVyZmxvdy15XCJdLFxuICAgICAgICBzdWJ0cmVlOiB0cnVlXG4gICAgICB9KTtcbiAgICB9XG4gIH1cblxufVxuIl19
|
155
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable-host.service.js","sourceRoot":"","sources":["../../../../../projects/core/src/modules/scrollable-host/scrollable-host.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { ElementRef, Injectable } from \"@angular/core\";\nimport { fromEvent, Observable, Subject, Subscriber, Subscription } from \"rxjs\";\nimport { takeUntil } from \"rxjs/operators\";\nimport { MutationObserverService } from \"../mutation/mutation-observer-service\";\nimport { SkyAppWindowRef } from \"../window/window-ref\";\n\nfunction notifySubscribers(subscribers: Subscriber<unknown>[], item?: unknown) {\n  for (const subscriber of subscribers) {\n    subscriber.next(item);\n  }\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class SkyScrollableHostService {\n\n  constructor(\n    private mutationObserverSvc: MutationObserverService,\n    private windowRef: SkyAppWindowRef\n  ) { }\n\n  /**\n   * Returns the given element's current scrollable host\n   * @param elementRef The element whose scrollable host is being requested\n   * @returns The current scrollable host\n   */\n  public getScrollableHost(elementRef: ElementRef): HTMLElement | Window {\n    return this.findScrollableHost(elementRef.nativeElement);\n  }\n\n  /**\n   * Returns an observable which emits the given element's current scrollable host\n   * @param elementRef The element whose scrollable host is being requested\n   * @param completionObservable An observable which alerts the internal observers that they should complete\n   * @returns An observable which emits the current scrollable host\n   * @internal\n   */\n  public watchScrollableHost(elementRef: ElementRef): Observable<HTMLElement | Window> {\n    let subscribers: Subscriber<HTMLElement | Window>[] = [];\n    let mutationObserver: MutationObserver;\n\n    return new Observable((subscriber) => {\n      subscribers.push(subscriber);\n\n      let scrollableHost = this.findScrollableHost(elementRef.nativeElement);\n      if (subscribers.length === 1) {\n\n        mutationObserver = this.mutationObserverSvc.create(() => {\n          let newScrollableHost = this.findScrollableHost(elementRef.nativeElement);\n\n          if (newScrollableHost !== scrollableHost) {\n            scrollableHost = newScrollableHost;\n            this.observeForScrollableHostChanges(scrollableHost, mutationObserver);\n\n            notifySubscribers(subscribers, scrollableHost);\n          }\n        });\n        this.observeForScrollableHostChanges(scrollableHost, mutationObserver);\n      }\n      subscriber.next(scrollableHost);\n\n      subscriber.add(() => {\n        const subIndex = subscribers.indexOf(subscriber);\n\n        /* sanity check */\n        /* istanbul ignore else */\n        if (subIndex >= 0) {\n          subscribers.splice(subIndex, 1);\n        }\n\n        if (subscribers.length === 0) {\n          mutationObserver.disconnect();\n        }\n      })\n    });\n  }\n\n  /**\n   * Returns an observable which emits whenever the element's scrollable host emits a scroll event. The observable will always emit the scroll events from the elements current scrollable host and will update based on any scrollable host changes. The observable will also emit once whenever the scrollable host changes.\n   * @param elementRef The element whose scrollable host scroll events are being requested\n   * @param completionObservable An observable which alerts the internal observers that they should complete\n   * @returns An observable which emits the scroll events from the given element's scrollable host\n   */\n  public watchScrollableHostScrollEvents(elementRef: ElementRef): Observable<void> {\n    let subscribers: Subscriber<void>[] = [];\n\n    let newScrollableHostObservable = new Subject();\n    let scrollableHostSubscription: Subscription;\n    let scrollEventSubscription: Subscription;\n    return new Observable((subscriber) => {\n      subscribers.push(subscriber);\n\n      if (subscribers.length === 1) {\n        scrollableHostSubscription = this.watchScrollableHost(elementRef)\n          .subscribe((scrollableHost) => {\n            newScrollableHostObservable.next();\n            newScrollableHostObservable.complete();\n            newScrollableHostObservable = new Subject();\n            scrollEventSubscription = fromEvent(scrollableHost, 'scroll')\n              .pipe(\n                takeUntil(newScrollableHostObservable)\n              )\n              .subscribe(() => {\n                notifySubscribers(subscribers);\n              });\n          });\n      }\n\n      subscriber.add(() => {\n        const subIndex = subscribers.indexOf(subscriber);\n\n        /* sanity check */\n        /* istanbul ignore else */\n        if (subIndex >= 0) {\n          subscribers.splice(subIndex, 1);\n        }\n\n        if (subscribers.length === 0) {\n          scrollableHostSubscription.unsubscribe();\n          scrollEventSubscription.unsubscribe();\n          newScrollableHostObservable.complete();\n        }\n      })\n    });\n  }\n\n  private findScrollableHost(element: HTMLElement): HTMLElement | Window {\n    const regex = /(auto|scroll)/;\n    const windowObj = this.windowRef.nativeWindow;\n    const bodyObj = windowObj.document.body;\n\n    /* Sanity check */\n    if (!element) {\n      return windowObj;\n    }\n\n    let style = windowObj.getComputedStyle(element);\n    let parent: HTMLElement = element;\n\n    do {\n      parent = <HTMLElement>parent.parentNode;\n\n      /* Sanity check for if this function is called for an element which has been removed from the DOM */\n      if (!(parent instanceof HTMLElement)) {\n        return windowObj\n      }\n\n      style = windowObj.getComputedStyle(parent);\n    } while (\n      !regex.test(style.overflow) &&\n      !regex.test(style.overflowY) &&\n      parent !== bodyObj\n    );\n\n    if (parent === bodyObj) {\n      return windowObj;\n    }\n\n    return parent;\n  }\n\n  private observeForScrollableHostChanges(element: HTMLElement | Window, mutationObserver: MutationObserver) {\n    mutationObserver.disconnect();\n    if (element instanceof HTMLElement) {\n      mutationObserver.observe(element, {\n        attributes: true,\n        attributeFilter: [\"class\", \"style.overflow\", \"style.overflow-y\"],\n        subtree: true\n      });\n    } else {\n      mutationObserver.observe(document.documentElement, {\n        attributes: true,\n        attributeFilter: [\"class\", \"style.overflow\", \"style.overflow-y\"],\n        subtree: true\n      });\n    }\n  }\n\n}\n"]}
|
@@ -75,7 +75,7 @@ export class SkyViewkeeperDirective {
|
|
75
75
|
else {
|
76
76
|
this.scrollableHostWatchUnsubscribe = new Subject();
|
77
77
|
}
|
78
|
-
this.scrollableHostService.watchScrollableHost(this.el
|
78
|
+
this.scrollableHostService.watchScrollableHost(this.el)
|
79
79
|
.pipe(takeUntil(this.scrollableHostWatchUnsubscribe))
|
80
80
|
.subscribe(scrollableHost => {
|
81
81
|
this.destroyViewkeepers();
|
@@ -107,4 +107,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.13", ngImpo
|
|
107
107
|
}] }]; }, propDecorators: { skyViewkeeper: [{
|
108
108
|
type: Input
|
109
109
|
}] } });
|
110
|
-
//# sourceMappingURL=data:application/json;base64,
|
110
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"viewkeeper.directive.js","sourceRoot":"","sources":["../../../../../projects/core/src/modules/viewkeeper/viewkeeper.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  ElementRef,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional\n} from '@angular/core';\nimport { Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\n\nimport {\n  MutationObserverService\n} from '../mutation/mutation-observer-service';\nimport { SkyScrollableHostService } from '../scrollable-host/scrollable-host.service';\n\nimport {\n  SkyViewkeeper\n} from './viewkeeper';\n\nimport {\n  SkyViewkeeperService\n} from './viewkeeper.service';\n\n@Directive({\n  selector: '[skyViewkeeper]'\n})\nexport class SkyViewkeeperDirective implements OnInit, OnDestroy {\n\n  @Input()\n  public set skyViewkeeper(value: string[]) {\n    this._skyViewkeeper = value;\n\n    this.detectElements();\n  }\n\n  public get skyViewkeeper(): string[] {\n    return this._skyViewkeeper;\n  }\n\n  private _skyViewkeeper: string[];\n\n  private viewkeepers: SkyViewkeeper[] = [];\n\n  private observer: MutationObserver;\n\n  private currentViewkeeperEls: HTMLElement[];\n\n  private scrollableHostWatchUnsubscribe: Subject<void> | undefined = undefined;\n\n  constructor(\n    private el: ElementRef,\n    private mutationObserverSvc: MutationObserverService,\n    private viewkeeperSvc: SkyViewkeeperService,\n    @Optional() private scrollableHostService: SkyScrollableHostService\n  ) { }\n\n  public ngOnInit(): void {\n    this.observer = this.mutationObserverSvc.create(() => this.detectElements());\n\n    this.observer.observe(\n      this.el.nativeElement,\n      {\n        childList: true,\n        subtree: true\n      }\n    );\n  }\n\n  public ngOnDestroy(): void {\n    this.observer.disconnect();\n\n    this.destroyViewkeepers();\n  }\n\n  private destroyViewkeepers(): void {\n    for (const viewkeeper of this.viewkeepers) {\n      this.viewkeeperSvc.destroy(viewkeeper);\n    }\n\n    this.viewkeepers = [];\n  }\n\n  private getViewkeeperEls(): HTMLElement[] {\n    let viewkeeperEls: HTMLElement[];\n\n    if (this.skyViewkeeper) {\n      viewkeeperEls = [];\n\n      for (const item of this.skyViewkeeper) {\n        let matchingEls = Array.from(\n          (this.el.nativeElement as HTMLElement).querySelectorAll(item)\n        ) as HTMLElement[];\n\n        viewkeeperEls = [...viewkeeperEls, ...matchingEls];\n      }\n    }\n\n    return viewkeeperEls;\n  }\n\n  private viewkeeperElsChanged(viewkeeperEls: HTMLElement[]): boolean {\n    if (!viewkeeperEls !== !this.currentViewkeeperEls) {\n      return true;\n    }\n\n    if (viewkeeperEls && this.currentViewkeeperEls) {\n      if (viewkeeperEls.length !== this.currentViewkeeperEls.length) {\n        return true;\n      }\n\n      for (let i = 0, n = viewkeeperEls.length; i < n; i++) {\n        if (viewkeeperEls[i] !== this.currentViewkeeperEls[i]) {\n          return true;\n        }\n      }\n    }\n\n    return false;\n  }\n\n  private detectElements(): void {\n    let viewkeeperEls = this.getViewkeeperEls();\n\n    if (this.viewkeeperElsChanged(viewkeeperEls)) {\n\n      if (this.scrollableHostWatchUnsubscribe) {\n        this.scrollableHostWatchUnsubscribe.next();\n        this.scrollableHostWatchUnsubscribe = new Subject();\n      } else {\n        this.scrollableHostWatchUnsubscribe = new Subject();\n      }\n\n      this.scrollableHostService.watchScrollableHost(this.el)\n        .pipe(takeUntil(this.scrollableHostWatchUnsubscribe))\n        .subscribe(scrollableHost => {\n          this.destroyViewkeepers();\n\n          let previousViewkeeperEl: HTMLElement;\n\n          for (const viewkeeperEl of viewkeeperEls) {\n            this.viewkeepers.push(\n              this.viewkeeperSvc.create(\n                {\n                  boundaryEl: this.el.nativeElement,\n                  scrollableHost: scrollableHost instanceof HTMLElement ? scrollableHost : undefined,\n                  el: viewkeeperEl,\n                  setWidth: true,\n                  verticalOffsetEl: previousViewkeeperEl\n                }\n              )\n            );\n\n            previousViewkeeperEl = viewkeeperEl;\n          }\n        });\n\n\n      this.currentViewkeeperEls = viewkeeperEls;\n    }\n  }\n}\n"]}
|
package/fesm2015/skyux-core.js
CHANGED
@@ -2,8 +2,8 @@ import * as i0 from '@angular/core';
|
|
2
2
|
import { NgModule, Injectable, EventEmitter, Directive, Input, Output, Injector, ViewContainerRef, Component, ChangeDetectionStrategy, ViewChild, Pipe, ElementRef, Optional } from '@angular/core';
|
3
3
|
import * as i4 from '@angular/common';
|
4
4
|
import { CommonModule } from '@angular/common';
|
5
|
-
import { Subject, fromEvent, BehaviorSubject, of } from 'rxjs';
|
6
|
-
import { takeUntil, debounceTime
|
5
|
+
import { Subject, fromEvent, BehaviorSubject, Observable, of } from 'rxjs';
|
6
|
+
import { takeUntil, debounceTime } from 'rxjs/operators';
|
7
7
|
import * as i1 from '@skyux/i18n';
|
8
8
|
import { getLibStringForLocale, SkyI18nModule, SKY_LIB_RESOURCES_PROVIDERS, SkyIntlNumberFormatStyle, SkyIntlNumberFormatter } from '@skyux/i18n';
|
9
9
|
import * as i3 from '@angular/router';
|
@@ -2456,30 +2456,102 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.13", ngImpo
|
|
2456
2456
|
}]
|
2457
2457
|
}] });
|
2458
2458
|
|
2459
|
+
function notifySubscribers(subscribers, item) {
|
2460
|
+
for (const subscriber of subscribers) {
|
2461
|
+
subscriber.next(item);
|
2462
|
+
}
|
2463
|
+
}
|
2459
2464
|
class SkyScrollableHostService {
|
2460
2465
|
constructor(mutationObserverSvc, windowRef) {
|
2461
2466
|
this.mutationObserverSvc = mutationObserverSvc;
|
2462
2467
|
this.windowRef = windowRef;
|
2463
2468
|
}
|
2464
|
-
|
2469
|
+
/**
|
2470
|
+
* Returns the given element's current scrollable host
|
2471
|
+
* @param elementRef The element whose scrollable host is being requested
|
2472
|
+
* @returns The current scrollable host
|
2473
|
+
*/
|
2474
|
+
getScrollableHost(elementRef) {
|
2465
2475
|
return this.findScrollableHost(elementRef.nativeElement);
|
2466
2476
|
}
|
2467
|
-
|
2468
|
-
|
2469
|
-
|
2470
|
-
|
2471
|
-
|
2472
|
-
|
2473
|
-
|
2477
|
+
/**
|
2478
|
+
* Returns an observable which emits the given element's current scrollable host
|
2479
|
+
* @param elementRef The element whose scrollable host is being requested
|
2480
|
+
* @param completionObservable An observable which alerts the internal observers that they should complete
|
2481
|
+
* @returns An observable which emits the current scrollable host
|
2482
|
+
* @internal
|
2483
|
+
*/
|
2484
|
+
watchScrollableHost(elementRef) {
|
2485
|
+
let subscribers = [];
|
2486
|
+
let mutationObserver;
|
2487
|
+
return new Observable((subscriber) => {
|
2488
|
+
subscribers.push(subscriber);
|
2489
|
+
let scrollableHost = this.findScrollableHost(elementRef.nativeElement);
|
2490
|
+
if (subscribers.length === 1) {
|
2491
|
+
mutationObserver = this.mutationObserverSvc.create(() => {
|
2492
|
+
let newScrollableHost = this.findScrollableHost(elementRef.nativeElement);
|
2493
|
+
if (newScrollableHost !== scrollableHost) {
|
2494
|
+
scrollableHost = newScrollableHost;
|
2495
|
+
this.observeForScrollableHostChanges(scrollableHost, mutationObserver);
|
2496
|
+
notifySubscribers(subscribers, scrollableHost);
|
2497
|
+
}
|
2498
|
+
});
|
2474
2499
|
this.observeForScrollableHostChanges(scrollableHost, mutationObserver);
|
2475
|
-
behaviorSubject.next(scrollableHost);
|
2476
2500
|
}
|
2501
|
+
subscriber.next(scrollableHost);
|
2502
|
+
subscriber.add(() => {
|
2503
|
+
const subIndex = subscribers.indexOf(subscriber);
|
2504
|
+
/* sanity check */
|
2505
|
+
/* istanbul ignore else */
|
2506
|
+
if (subIndex >= 0) {
|
2507
|
+
subscribers.splice(subIndex, 1);
|
2508
|
+
}
|
2509
|
+
if (subscribers.length === 0) {
|
2510
|
+
mutationObserver.disconnect();
|
2511
|
+
}
|
2512
|
+
});
|
2477
2513
|
});
|
2478
|
-
|
2479
|
-
|
2480
|
-
|
2514
|
+
}
|
2515
|
+
/**
|
2516
|
+
* Returns an observable which emits whenever the element's scrollable host emits a scroll event. The observable will always emit the scroll events from the elements current scrollable host and will update based on any scrollable host changes. The observable will also emit once whenever the scrollable host changes.
|
2517
|
+
* @param elementRef The element whose scrollable host scroll events are being requested
|
2518
|
+
* @param completionObservable An observable which alerts the internal observers that they should complete
|
2519
|
+
* @returns An observable which emits the scroll events from the given element's scrollable host
|
2520
|
+
*/
|
2521
|
+
watchScrollableHostScrollEvents(elementRef) {
|
2522
|
+
let subscribers = [];
|
2523
|
+
let newScrollableHostObservable = new Subject();
|
2524
|
+
let scrollableHostSubscription;
|
2525
|
+
let scrollEventSubscription;
|
2526
|
+
return new Observable((subscriber) => {
|
2527
|
+
subscribers.push(subscriber);
|
2528
|
+
if (subscribers.length === 1) {
|
2529
|
+
scrollableHostSubscription = this.watchScrollableHost(elementRef)
|
2530
|
+
.subscribe((scrollableHost) => {
|
2531
|
+
newScrollableHostObservable.next();
|
2532
|
+
newScrollableHostObservable.complete();
|
2533
|
+
newScrollableHostObservable = new Subject();
|
2534
|
+
scrollEventSubscription = fromEvent(scrollableHost, 'scroll')
|
2535
|
+
.pipe(takeUntil(newScrollableHostObservable))
|
2536
|
+
.subscribe(() => {
|
2537
|
+
notifySubscribers(subscribers);
|
2538
|
+
});
|
2539
|
+
});
|
2540
|
+
}
|
2541
|
+
subscriber.add(() => {
|
2542
|
+
const subIndex = subscribers.indexOf(subscriber);
|
2543
|
+
/* sanity check */
|
2544
|
+
/* istanbul ignore else */
|
2545
|
+
if (subIndex >= 0) {
|
2546
|
+
subscribers.splice(subIndex, 1);
|
2547
|
+
}
|
2548
|
+
if (subscribers.length === 0) {
|
2549
|
+
scrollableHostSubscription.unsubscribe();
|
2550
|
+
scrollEventSubscription.unsubscribe();
|
2551
|
+
newScrollableHostObservable.complete();
|
2552
|
+
}
|
2553
|
+
});
|
2481
2554
|
});
|
2482
|
-
return behaviorSubject;
|
2483
2555
|
}
|
2484
2556
|
findScrollableHost(element) {
|
2485
2557
|
const regex = /(auto|scroll)/;
|
@@ -2923,7 +2995,7 @@ class SkyViewkeeperDirective {
|
|
2923
2995
|
else {
|
2924
2996
|
this.scrollableHostWatchUnsubscribe = new Subject();
|
2925
2997
|
}
|
2926
|
-
this.scrollableHostService.watchScrollableHost(this.el
|
2998
|
+
this.scrollableHostService.watchScrollableHost(this.el)
|
2927
2999
|
.pipe(takeUntil(this.scrollableHostWatchUnsubscribe))
|
2928
3000
|
.subscribe(scrollableHost => {
|
2929
3001
|
this.destroyViewkeepers();
|