@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.
@@ -1,33 +1,105 @@
1
1
  import { Injectable } from "@angular/core";
2
- import { BehaviorSubject } from "rxjs";
3
- import { take } from "rxjs/operators";
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
- getScrollabeHost(elementRef) {
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
- watchScrollableHost(elementRef, completionObservable) {
16
- let scrollableHost = this.findScrollableHost(elementRef.nativeElement);
17
- let behaviorSubject = new BehaviorSubject(scrollableHost);
18
- const mutationObserver = this.mutationObserverSvc.create(() => {
19
- let newScrollableHost = this.findScrollableHost(elementRef.nativeElement);
20
- if (newScrollableHost !== scrollableHost) {
21
- scrollableHost = newScrollableHost;
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
- this.observeForScrollableHostChanges(scrollableHost, mutationObserver);
27
- completionObservable.pipe(take(1)).subscribe(() => {
28
- mutationObserver.disconnect();
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, this.scrollableHostWatchUnsubscribe)
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,{"version":3,"file":"viewkeeper.directive.js","sourceRoot":"","sources":["../../../../../projects/core/src/modules/viewkeeper/viewkeeper.directive.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EAET,KAAK,EAGL,QAAQ,EACT,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAC/B,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;;;;;AAkB3C,MAAM,OAAO,sBAAsB;IAuBjC,YACU,EAAc,EACd,mBAA4C,EAC5C,aAAmC,EACvB,qBAA+C;QAH3D,OAAE,GAAF,EAAE,CAAY;QACd,wBAAmB,GAAnB,mBAAmB,CAAyB;QAC5C,kBAAa,GAAb,aAAa,CAAsB;QACvB,0BAAqB,GAArB,qBAAqB,CAA0B;QAZ7D,gBAAW,GAAoB,EAAE,CAAC;QAMlC,mCAA8B,GAA8B,SAAS,CAAC;IAO1E,CAAC;IA1BL,IACW,aAAa,CAAC,KAAe;QACtC,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;QAE5B,IAAI,CAAC,cAAc,EAAE,CAAC;IACxB,CAAC;IAED,IAAW,aAAa;QACtB,OAAO,IAAI,CAAC,cAAc,CAAC;IAC7B,CAAC;IAmBM,QAAQ;QACb,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;QAE7E,IAAI,CAAC,QAAQ,CAAC,OAAO,CACnB,IAAI,CAAC,EAAE,CAAC,aAAa,EACrB;YACE,SAAS,EAAE,IAAI;YACf,OAAO,EAAE,IAAI;SACd,CACF,CAAC;IACJ,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC;QAE3B,IAAI,CAAC,kBAAkB,EAAE,CAAC;IAC5B,CAAC;IAEO,kBAAkB;QACxB,KAAK,MAAM,UAAU,IAAI,IAAI,CAAC,WAAW,EAAE;YACzC,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;SACxC;QAED,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC;IACxB,CAAC;IAEO,gBAAgB;QACtB,IAAI,aAA4B,CAAC;QAEjC,IAAI,IAAI,CAAC,aAAa,EAAE;YACtB,aAAa,GAAG,EAAE,CAAC;YAEnB,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,aAAa,EAAE;gBACrC,IAAI,WAAW,GAAG,KAAK,CAAC,IAAI,CACzB,IAAI,CAAC,EAAE,CAAC,aAA6B,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAC7C,CAAC;gBAEnB,aAAa,GAAG,CAAC,GAAG,aAAa,EAAE,GAAG,WAAW,CAAC,CAAC;aACpD;SACF;QAED,OAAO,aAAa,CAAC;IACvB,CAAC;IAEO,oBAAoB,CAAC,aAA4B;QACvD,IAAI,CAAC,aAAa,KAAK,CAAC,IAAI,CAAC,oBAAoB,EAAE;YACjD,OAAO,IAAI,CAAC;SACb;QAED,IAAI,aAAa,IAAI,IAAI,CAAC,oBAAoB,EAAE;YAC9C,IAAI,aAAa,CAAC,MAAM,KAAK,IAAI,CAAC,oBAAoB,CAAC,MAAM,EAAE;gBAC7D,OAAO,IAAI,CAAC;aACb;YAED,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,aAAa,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE;gBACpD,IAAI,aAAa,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,EAAE;oBACrD,OAAO,IAAI,CAAC;iBACb;aACF;SACF;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,cAAc;QACpB,IAAI,aAAa,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAE5C,IAAI,IAAI,CAAC,oBAAoB,CAAC,aAAa,CAAC,EAAE;YAE5C,IAAI,IAAI,CAAC,8BAA8B,EAAE;gBACvC,IAAI,CAAC,8BAA8B,CAAC,IAAI,EAAE,CAAC;gBAC3C,IAAI,CAAC,8BAA8B,GAAG,IAAI,OAAO,EAAE,CAAC;aACrD;iBAAM;gBACL,IAAI,CAAC,8BAA8B,GAAG,IAAI,OAAO,EAAE,CAAC;aACrD;YAED,IAAI,CAAC,qBAAqB,CAAC,mBAAmB,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,CAAC,8BAA8B,CAAC;iBACzF,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;iBACpD,SAAS,CAAC,cAAc,CAAC,EAAE;gBAC1B,IAAI,CAAC,kBAAkB,EAAE,CAAC;gBAE1B,IAAI,oBAAiC,CAAC;gBAEtC,KAAK,MAAM,YAAY,IAAI,aAAa,EAAE;oBACxC,IAAI,CAAC,WAAW,CAAC,IAAI,CACnB,IAAI,CAAC,aAAa,CAAC,MAAM,CACvB;wBACE,UAAU,EAAE,IAAI,CAAC,EAAE,CAAC,aAAa;wBACjC,cAAc,EAAE,cAAc,YAAY,WAAW,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,SAAS;wBAClF,EAAE,EAAE,YAAY;wBAChB,QAAQ,EAAE,IAAI;wBACd,gBAAgB,EAAE,oBAAoB;qBACvC,CACF,CACF,CAAC;oBAEF,oBAAoB,GAAG,YAAY,CAAC;iBACrC;YACH,CAAC,CAAC,CAAC;YAGL,IAAI,CAAC,oBAAoB,GAAG,aAAa,CAAC;SAC3C;IACH,CAAC;;oHArIU,sBAAsB;wGAAtB,sBAAsB;4FAAtB,sBAAsB;kBAHlC,SAAS;mBAAC;oBACT,QAAQ,EAAE,iBAAiB;iBAC5B;;0BA4BI,QAAQ;4CAxBA,aAAa;sBADvB,KAAK","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, this.scrollableHostWatchUnsubscribe)\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"]}
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"]}
@@ -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, take } from 'rxjs/operators';
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
- getScrollabeHost(elementRef) {
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
- watchScrollableHost(elementRef, completionObservable) {
2468
- let scrollableHost = this.findScrollableHost(elementRef.nativeElement);
2469
- let behaviorSubject = new BehaviorSubject(scrollableHost);
2470
- const mutationObserver = this.mutationObserverSvc.create(() => {
2471
- let newScrollableHost = this.findScrollableHost(elementRef.nativeElement);
2472
- if (newScrollableHost !== scrollableHost) {
2473
- scrollableHost = newScrollableHost;
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
- this.observeForScrollableHostChanges(scrollableHost, mutationObserver);
2479
- completionObservable.pipe(take(1)).subscribe(() => {
2480
- mutationObserver.disconnect();
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, this.scrollableHostWatchUnsubscribe)
2998
+ this.scrollableHostService.watchScrollableHost(this.el)
2927
2999
  .pipe(takeUntil(this.scrollableHostWatchUnsubscribe))
2928
3000
  .subscribe(scrollableHost => {
2929
3001
  this.destroyViewkeepers();