@ngutil/layout 0.0.3-dev.7 → 0.0.3-dev.9

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.
@@ -2,7 +2,7 @@ import { inject, NgZone } from "@angular/core";
2
2
  import { distinctUntilChanged, Observable, shareReplay } from "rxjs";
3
3
  import { NumberWithUnit } from "@ngutil/common";
4
4
  const RESIZE_WATCHES = new Map();
5
- const SCROLL_CONTENT_WATCHES = new Map();
5
+ const SCROLL_WATCHES = new Map();
6
6
  export function watchDimension(el, box = "border-box") {
7
7
  const zone = inject(NgZone);
8
8
  return box === "scroll-box" ? _watchScroll(zone, el) : _watchResize(zone, el, box);
@@ -11,7 +11,7 @@ function _watchResize(zone, el, box) {
11
11
  return _watch(zone, el, RESIZE_WATCHES, () => _createResizeWatcher(zone, el, box));
12
12
  }
13
13
  function _watchScroll(zone, el) {
14
- return _watch(zone, el, SCROLL_CONTENT_WATCHES, () => _createScollWatcher(zone, el));
14
+ return _watch(zone, el, SCROLL_WATCHES, () => _createScollWatcher(zone, el));
15
15
  }
16
16
  function _watch(zone, el, watches, factory) {
17
17
  const existing = watches.get(el);
@@ -74,10 +74,11 @@ function _createScollWatcher(zone, el) {
74
74
  return () => {
75
75
  dimSum.unsubscribe();
76
76
  mutation.disconnect();
77
+ SCROLL_WATCHES.delete(el);
77
78
  };
78
79
  }).pipe(shareReplay(1)));
79
80
  }
80
81
  function _number(val) {
81
82
  return new NumberWithUnit(val, "pk");
82
83
  }
83
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dimension-watcher.js","sourceRoot":"","sources":["../../../../../packages/layout/src/util/dimension-watcher.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, NgZone } from \"@angular/core\"\n\nimport { distinctUntilChanged, Observable, shareReplay, Subscriber } from \"rxjs\"\n\nimport { NumberWithUnit } from \"@ngutil/common\"\n\nimport { Dimension } from \"./dimension\"\n\ntype Watches = Map<HTMLElement, Observable<Dimension>>\nexport type WatchBox = ResizeObserverBoxOptions | \"scroll-box\"\n\nconst RESIZE_WATCHES: Watches = new Map()\nconst SCROLL_CONTENT_WATCHES: Watches = new Map()\n\nexport function watchDimension(el: HTMLElement, box: WatchBox = \"border-box\"): Observable<Dimension> {\n    const zone = inject(NgZone)\n    return box === \"scroll-box\" ? _watchScroll(zone, el) : _watchResize(zone, el, box)\n}\n\nfunction _watchResize(zone: NgZone, el: HTMLElement, box: WatchBox) {\n    return _watch(zone, el, RESIZE_WATCHES, () => _createResizeWatcher(zone, el, box))\n}\n\nfunction _watchScroll(zone: NgZone, el: HTMLElement) {\n    return _watch(zone, el, SCROLL_CONTENT_WATCHES, () => _createScollWatcher(zone, el))\n}\n\nfunction _watch(zone: NgZone, el: HTMLElement, watches: Watches, factory: () => Observable<Dimension>) {\n    const existing = watches.get(el)\n    if (existing == null) {\n        const watcher = factory()\n        watches.set(el, watcher)\n        return watcher\n    }\n    return existing\n}\n\nfunction _createResizeWatcher(zone: NgZone, el: HTMLElement, box: WatchBox): Observable<Dimension> {\n    if (box !== \"border-box\") {\n        throw new Error(`Currently not implemented box mode: ${box}`)\n    }\n\n    return zone.runOutsideAngular(() =>\n        new Observable((sub: Subscriber<Dimension>) => {\n            const observer = new ResizeObserver(entries => {\n                for (const entry of entries) {\n                    if (entry.borderBoxSize) {\n                        sub.next({\n                            width: _number(entry.borderBoxSize[0].inlineSize),\n                            height: _number(entry.borderBoxSize[0].blockSize)\n                        })\n                    } else {\n                        sub.next({\n                            width: _number(el.offsetWidth),\n                            height: _number(el.offsetHeight)\n                        })\n                    }\n                }\n            })\n            observer.observe(el, { box: box as ResizeObserverBoxOptions })\n\n            return () => {\n                observer.disconnect()\n                RESIZE_WATCHES.delete(el)\n            }\n        }).pipe(\n            distinctUntilChanged((p, c) => p && c && p.width === c.width && p.height === c.height),\n            shareReplay(1)\n        )\n    )\n}\n\nfunction _createScollWatcher(zone: NgZone, el: HTMLElement): Observable<Dimension> {\n    return zone.runOutsideAngular(() =>\n        new Observable((sub: Subscriber<Dimension>) => {\n            let lastSw: number = 0\n            let lastSh: number = 0\n\n            const emit = () => {\n                const sw = el.scrollWidth\n                const sh = el.scrollHeight\n                if (lastSw !== sw || lastSh !== sh) {\n                    lastSw = sw\n                    lastSh = sh\n                    sub.next({ width: _number(lastSw), height: _number(lastSh) })\n                }\n            }\n\n            const dimSum = _watchResize(zone, el, \"border-box\").subscribe(emit)\n            const mutation = new MutationObserver(emit)\n            mutation.observe(el, {\n                subtree: true,\n                childList: true,\n                attributes: true,\n                characterData: true\n            })\n\n            return () => {\n                dimSum.unsubscribe()\n                mutation.disconnect()\n            }\n        }).pipe(shareReplay(1))\n    )\n}\n\nfunction _number(val: number): NumberWithUnit {\n    return new NumberWithUnit(val, \"pk\")\n}\n"]}
84
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dimension-watcher.js","sourceRoot":"","sources":["../../../../../packages/layout/src/util/dimension-watcher.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, NgZone } from \"@angular/core\"\n\nimport { distinctUntilChanged, Observable, shareReplay, Subscriber } from \"rxjs\"\n\nimport { NumberWithUnit } from \"@ngutil/common\"\n\nimport { Dimension } from \"./dimension\"\n\ntype Watches = Map<HTMLElement, Observable<Dimension>>\nexport type WatchBox = ResizeObserverBoxOptions | \"scroll-box\"\n\nconst RESIZE_WATCHES: Watches = new Map()\nconst SCROLL_WATCHES: Watches = new Map()\n\nexport function watchDimension(el: HTMLElement, box: WatchBox = \"border-box\"): Observable<Dimension> {\n    const zone = inject(NgZone)\n    return box === \"scroll-box\" ? _watchScroll(zone, el) : _watchResize(zone, el, box)\n}\n\nfunction _watchResize(zone: NgZone, el: HTMLElement, box: WatchBox) {\n    return _watch(zone, el, RESIZE_WATCHES, () => _createResizeWatcher(zone, el, box))\n}\n\nfunction _watchScroll(zone: NgZone, el: HTMLElement) {\n    return _watch(zone, el, SCROLL_WATCHES, () => _createScollWatcher(zone, el))\n}\n\nfunction _watch(zone: NgZone, el: HTMLElement, watches: Watches, factory: () => Observable<Dimension>) {\n    const existing = watches.get(el)\n    if (existing == null) {\n        const watcher = factory()\n        watches.set(el, watcher)\n        return watcher\n    }\n    return existing\n}\n\nfunction _createResizeWatcher(zone: NgZone, el: HTMLElement, box: WatchBox): Observable<Dimension> {\n    if (box !== \"border-box\") {\n        throw new Error(`Currently not implemented box mode: ${box}`)\n    }\n\n    return zone.runOutsideAngular(() =>\n        new Observable((sub: Subscriber<Dimension>) => {\n            const observer = new ResizeObserver(entries => {\n                for (const entry of entries) {\n                    if (entry.borderBoxSize) {\n                        sub.next({\n                            width: _number(entry.borderBoxSize[0].inlineSize),\n                            height: _number(entry.borderBoxSize[0].blockSize)\n                        })\n                    } else {\n                        sub.next({\n                            width: _number(el.offsetWidth),\n                            height: _number(el.offsetHeight)\n                        })\n                    }\n                }\n            })\n            observer.observe(el, { box: box as ResizeObserverBoxOptions })\n\n            return () => {\n                observer.disconnect()\n                RESIZE_WATCHES.delete(el)\n            }\n        }).pipe(\n            distinctUntilChanged((p, c) => p && c && p.width === c.width && p.height === c.height),\n            shareReplay(1)\n        )\n    )\n}\n\nfunction _createScollWatcher(zone: NgZone, el: HTMLElement): Observable<Dimension> {\n    return zone.runOutsideAngular(() =>\n        new Observable((sub: Subscriber<Dimension>) => {\n            let lastSw: number = 0\n            let lastSh: number = 0\n\n            const emit = () => {\n                const sw = el.scrollWidth\n                const sh = el.scrollHeight\n                if (lastSw !== sw || lastSh !== sh) {\n                    lastSw = sw\n                    lastSh = sh\n                    sub.next({ width: _number(lastSw), height: _number(lastSh) })\n                }\n            }\n\n            const dimSum = _watchResize(zone, el, \"border-box\").subscribe(emit)\n            const mutation = new MutationObserver(emit)\n            mutation.observe(el, {\n                subtree: true,\n                childList: true,\n                attributes: true,\n                characterData: true\n            })\n\n            return () => {\n                dimSum.unsubscribe()\n                mutation.disconnect()\n                SCROLL_WATCHES.delete(el)\n            }\n        }).pipe(shareReplay(1))\n    )\n}\n\nfunction _number(val: number): NumberWithUnit {\n    return new NumberWithUnit(val, \"pk\")\n}\n"]}
@@ -1,4 +1,3 @@
1
1
  export { watchMedia } from "./media-watcher";
2
2
  export { watchDimension } from "./dimension-watcher";
3
- export { FastDOM } from "./dom";
4
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9sYXlvdXQvc3JjL3V0aWwvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGlCQUFpQixDQUFBO0FBQzVDLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQTtBQUNwRCxPQUFPLEVBQUUsT0FBTyxFQUFFLE1BQU0sT0FBTyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgRGltZW5zaW9uIH0gZnJvbSBcIi4vZGltZW5zaW9uXCJcbmV4cG9ydCB7IFJlY3QgfSBmcm9tIFwiLi9yZWN0XCJcbmV4cG9ydCB7IHdhdGNoTWVkaWEgfSBmcm9tIFwiLi9tZWRpYS13YXRjaGVyXCJcbmV4cG9ydCB7IHdhdGNoRGltZW5zaW9uIH0gZnJvbSBcIi4vZGltZW5zaW9uLXdhdGNoZXJcIlxuZXhwb3J0IHsgRmFzdERPTSB9IGZyb20gXCIuL2RvbVwiXG4iXX0=
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9sYXlvdXQvc3JjL3V0aWwvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGlCQUFpQixDQUFBO0FBQzVDLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IERpbWVuc2lvbiB9IGZyb20gXCIuL2RpbWVuc2lvblwiXG5leHBvcnQgeyBSZWN0IH0gZnJvbSBcIi4vcmVjdFwiXG5leHBvcnQgeyB3YXRjaE1lZGlhIH0gZnJvbSBcIi4vbWVkaWEtd2F0Y2hlclwiXG5leHBvcnQgeyB3YXRjaERpbWVuc2lvbiB9IGZyb20gXCIuL2RpbWVuc2lvbi13YXRjaGVyXCJcbiJdfQ==
@@ -1,7 +1,7 @@
1
1
  import * as i0 from '@angular/core';
2
- import { inject, NgZone, Component, ElementRef, Input, Output, ChangeDetectionStrategy, ContentChild, ContentChildren, NgModule } from '@angular/core';
3
- import { Observable, distinctUntilChanged, shareReplay, BehaviorSubject, map, combineLatest, switchMap, of, Subject, startWith } from 'rxjs';
4
- import { NumberWithUnit, rawRequestAnimationFrame, Destructible, coerceBoolAttr } from '@ngutil/common';
2
+ import { inject, NgZone, Component, ElementRef, Input, Output, ChangeDetectionStrategy, ContentChild, ContentChildren, NgModule, Injectable, TemplateRef, Directive, ViewContainerRef, Injector } from '@angular/core';
3
+ import { Observable, distinctUntilChanged, shareReplay, BehaviorSubject, map, combineLatest, switchMap, of, Subject, startWith, scan, tap, finalize } from 'rxjs';
4
+ import { NumberWithUnit, Destructible, coerceBoolAttr, FastDOM } from '@ngutil/common';
5
5
 
6
6
  const WATCHES = {};
7
7
  /**
@@ -31,7 +31,7 @@ function _createWatcher(expr) {
31
31
  }
32
32
 
33
33
  const RESIZE_WATCHES = new Map();
34
- const SCROLL_CONTENT_WATCHES = new Map();
34
+ const SCROLL_WATCHES = new Map();
35
35
  function watchDimension(el, box = "border-box") {
36
36
  const zone = inject(NgZone);
37
37
  return box === "scroll-box" ? _watchScroll(zone, el) : _watchResize(zone, el, box);
@@ -40,7 +40,7 @@ function _watchResize(zone, el, box) {
40
40
  return _watch(zone, el, RESIZE_WATCHES, () => _createResizeWatcher(zone, el, box));
41
41
  }
42
42
  function _watchScroll(zone, el) {
43
- return _watch(zone, el, SCROLL_CONTENT_WATCHES, () => _createScollWatcher(zone, el));
43
+ return _watch(zone, el, SCROLL_WATCHES, () => _createScollWatcher(zone, el));
44
44
  }
45
45
  function _watch(zone, el, watches, factory) {
46
46
  const existing = watches.get(el);
@@ -103,6 +103,7 @@ function _createScollWatcher(zone, el) {
103
103
  return () => {
104
104
  dimSum.unsubscribe();
105
105
  mutation.disconnect();
106
+ SCROLL_WATCHES.delete(el);
106
107
  };
107
108
  }).pipe(shareReplay(1)));
108
109
  }
@@ -110,76 +111,6 @@ function _number(val) {
110
111
  return new NumberWithUnit(val, "pk");
111
112
  }
112
113
 
113
- class _FastDOM {
114
- #rafId;
115
- #mutate = [];
116
- #measure = [];
117
- mutate(handler) {
118
- this.#mutate.push(handler);
119
- this._schedule();
120
- }
121
- mutateNext(handler) {
122
- this.#mutate.push(() => {
123
- this.#mutate.push(handler);
124
- });
125
- this._schedule();
126
- }
127
- measure(handler) {
128
- this.#measure.push(handler);
129
- this._schedule();
130
- }
131
- setStyle(el, style, chain) {
132
- this.mutate(() => {
133
- for (const [k, v] of Object.entries(style)) {
134
- if (v == null) {
135
- el.style.removeProperty(k);
136
- }
137
- else {
138
- el.style.setProperty(k, v);
139
- }
140
- }
141
- chain && chain();
142
- });
143
- }
144
- setAttributes(el, attrs, chain) {
145
- this.mutate(() => {
146
- for (const [k, v] of Object.entries(attrs)) {
147
- if (v == null) {
148
- el.removeAttribute(k);
149
- }
150
- else {
151
- el.setAttribute(k, v);
152
- }
153
- }
154
- chain && chain();
155
- });
156
- }
157
- _schedule() {
158
- if (!this.#rafId) {
159
- this.#rafId = rawRequestAnimationFrame(this._apply.bind(this));
160
- }
161
- }
162
- _apply() {
163
- this.#rafId = null;
164
- const measure = this.#measure.slice();
165
- const mutate = this.#mutate.slice();
166
- this.#measure.length = 0;
167
- this.#mutate.length = 0;
168
- runQ(measure);
169
- runQ(mutate);
170
- if (this.#measure.length || this.#mutate.length) {
171
- this._schedule();
172
- }
173
- }
174
- }
175
- function runQ(items) {
176
- let item;
177
- while ((item = items.shift())) {
178
- item();
179
- }
180
- }
181
- const FastDOM = new _FastDOM();
182
-
183
114
  /**
184
115
  * -----------------------------------------------
185
116
  * | TOP:LEFT | TOP:CENTER | TOP:RIGHT |
@@ -329,7 +260,6 @@ const DEFAULT_POSITION = L9Range.coerce("left");
329
260
  class DockingPanelComponent extends Destructible {
330
261
  set positionInput(val) {
331
262
  const coerced = L9Range.coerce(val);
332
- console.log("SET POSITION", coerced, this.position.value.isEq(coerced));
333
263
  if (coerced.orient === "rect") {
334
264
  throw new Error(`Invalid position value: ${val}`);
335
265
  }
@@ -523,7 +453,6 @@ class DockingLayoutComponent extends Destructible {
523
453
  }
524
454
  }
525
455
  #layout(entries) {
526
- console.log("layout", entries);
527
456
  let paddingTop = 0;
528
457
  let paddingRight = 0;
529
458
  let paddingBottom = 0;
@@ -600,7 +529,6 @@ class DockingLayoutComponent extends Destructible {
600
529
  "--docking-panel-real-h": isHorizontal ? `${panelSize}px` : null
601
530
  });
602
531
  }
603
- console.log({ paddingTop, paddingRight, paddingBottom, paddingLeft });
604
532
  FastDOM.setStyle(this.#el.nativeElement, {
605
533
  "--docking-layout-top": `${paddingTop}px`,
606
534
  "--docking-layout-right": `${paddingRight}px`,
@@ -609,84 +537,6 @@ class DockingLayoutComponent extends Destructible {
609
537
  });
610
538
  }
611
539
  }
612
- #layoutOld(entries) {
613
- // let paddingTop = 0
614
- // let paddingRight = 0
615
- // let paddingBottom = 0
616
- // let paddingLeft = 0
617
- // if (!this.contentOnly) {
618
- // let embeddedZIndex = EMBEDDED_ZINDEX
619
- // let overlayZIndex = OVERLAY_ZINDEX
620
- // const leftRight: PanelRefChanges[] = entries.filter(v =>
621
- // ["left", "right"].includes(v.changes.position.side)
622
- // )
623
- // const topBottom: PanelRefChanges[] = entries.filter(v =>
624
- // ["top", "bottom"].includes(v.changes.position.side)
625
- // )
626
- // for (const entry of entries) {
627
- // const changes = entry.changes
628
- // const ref = entry.ref
629
- // if (changes.mode === "embedded") {
630
- // ref.style.zIndex = `${embeddedZIndex++}`
631
- // } else if (changes.mode === "overlay") {
632
- // ref.style.zIndex = `${overlayZIndex++}`
633
- // }
634
- // }
635
- // for (const entry of leftRight) {
636
- // const changes = entry.changes
637
- // const ref = entry.ref
638
- // const padding =
639
- // changes.mode === "embedded"
640
- // ? changes.state === "full"
641
- // ? changes.fullSize
642
- // : changes.state === "mini"
643
- // ? changes.miniSize
644
- // : 0
645
- // : 0
646
- // ref.style.top = "0"
647
- // ref.style.bottom = "0"
648
- // if (changes.position.side === "left") {
649
- // paddingLeft = Math.max(paddingLeft, padding)
650
- // ref.style.left = "0"
651
- // ref.style.right = ""
652
- // } else {
653
- // paddingRight = Math.max(paddingRight, padding)
654
- // ref.style.right = "0"
655
- // ref.style.left = ""
656
- // }
657
- // }
658
- // for (const entry of topBottom) {
659
- // const changes = entry.changes
660
- // const ref = entry.ref
661
- // const padding =
662
- // changes.mode === "embedded"
663
- // ? changes.state === "full"
664
- // ? changes.fullSize
665
- // : changes.state === "mini"
666
- // ? changes.miniSize
667
- // : 0
668
- // : 0
669
- // if (changes.mode === "embedded") {
670
- // ref.style.left = `${paddingLeft}px`
671
- // ref.style.right = `${paddingRight}px`
672
- // } else {
673
- // ref.style.left = "0"
674
- // ref.style.right = "0"
675
- // }
676
- // if (changes.position?.cells[0].v === "top") {
677
- // paddingTop = Math.max(paddingTop, padding)
678
- // ref.style.top = "0"
679
- // ref.style.bottom = ""
680
- // } else {
681
- // paddingBottom = Math.max(paddingBottom, padding)
682
- // ref.style.bottom = `0`
683
- // ref.style.top = ""
684
- // }
685
- // }
686
- // }
687
- // const cel = this.contentEl.nativeElement
688
- // cel.style.padding = `${paddingTop}px ${paddingRight}px ${paddingBottom}px ${paddingLeft}px`
689
- }
690
540
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: DockingLayoutComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
691
541
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.1.3", type: DockingLayoutComponent, isStandalone: true, selector: "nu-docking", inputs: { contentOnly: "contentOnly", positionMode: "positionMode" }, queries: [{ propertyName: "contentComponent", first: true, predicate: DockingContentComponent, descendants: true }, { propertyName: "dockingPanels", predicate: DockingPanelComponent }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: `
692
542
  <ng-content select="nu-docking-panel"></ng-content>
@@ -739,9 +589,245 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.3", ngImpor
739
589
  }]
740
590
  }] });
741
591
 
592
+ const SLOT_REGEX = /^([^:\s]+)(?::(\d+))?(?:\s+as\s+(.*?))?$/i;
593
+ class SlotDef {
594
+ constructor(slot, tpl) {
595
+ this.tpl = tpl;
596
+ const match = slot.match(SLOT_REGEX);
597
+ if (!match) {
598
+ console.warn(`Invalid slot definition: ${slot}`);
599
+ }
600
+ else {
601
+ this.slot = match[1];
602
+ this.order = match[2] != null ? Number(match[2]) : Infinity;
603
+ this.id = match[3];
604
+ }
605
+ }
606
+ dispose() {
607
+ this.viewRef?.destroy();
608
+ this.viewRef = undefined;
609
+ }
610
+ }
611
+ /**
612
+ * @Directive({selector: "ng-template[xyzSlot]", inputs: [{name: "slot", alias: "xyzSlot"}]})
613
+ * class XYZSlotDirective extends SlotDirective<XYZComponentSlots> { }
614
+ *
615
+ * @Directive({selector: "ng-template[xyzSlotOutlet]", inputs: [{name: "slot", alias: "xyzSlotOutlet"}]})
616
+ * class XYZSlotOutletDirective extends SlotOutletDirective<XYZComponentSlots> { }
617
+ *
618
+ *
619
+ * @Component({provides: [SlotsService]})
620
+ * class XYZComponent {
621
+ * slots: inject(SlotsService<XYZComponentSlots>)
622
+ * }
623
+ *
624
+ *
625
+ */
626
+ class SlotsService extends Destructible {
627
+ #events = new Subject();
628
+ #entries = this.#events.pipe(scan((entries, event) => {
629
+ if (event.type === "add") {
630
+ const index = entries.findIndex(value => value === event.def);
631
+ if (index > -1) {
632
+ entries[index] = event.def;
633
+ }
634
+ else {
635
+ entries.push(event.def);
636
+ }
637
+ }
638
+ else if (event.type === "del") {
639
+ const index = entries.findIndex(value => value === event.def);
640
+ if (index > -1) {
641
+ entries.splice(index, 1);
642
+ }
643
+ }
644
+ return entries;
645
+ }, []), tap(entries => {
646
+ entries.sort((a, b) => {
647
+ if (a.slot === b.slot) {
648
+ return a.order - b.order;
649
+ }
650
+ else {
651
+ return a.slot.localeCompare(b.slot);
652
+ }
653
+ });
654
+ }), shareReplay(1));
655
+ constructor() {
656
+ super();
657
+ // XXX: need to collect entries from the beginning
658
+ this.d.sub(this.#entries).subscribe();
659
+ }
660
+ addTpl(def) {
661
+ this.#events.next({ type: "add", def });
662
+ }
663
+ delTpl(def) {
664
+ this.#events.next({ type: "del", def });
665
+ }
666
+ #watchers = {};
667
+ watch(slot) {
668
+ const existing = this.#watchers[slot];
669
+ if (existing == null) {
670
+ return (this.#watchers[slot] = this.#watch(slot));
671
+ }
672
+ else {
673
+ return existing;
674
+ }
675
+ }
676
+ #watch(slot) {
677
+ return this.#entries.pipe(map(entries => entries.filter(entry => entry.slot === slot)), distinctUntilChanged((prev, curr) => {
678
+ if (prev.length === curr.length) {
679
+ for (let i = 0; i < prev.length; i++) {
680
+ if (prev[i] !== curr[i]) {
681
+ return false;
682
+ }
683
+ }
684
+ return true;
685
+ }
686
+ else {
687
+ return false;
688
+ }
689
+ }), finalize(() => {
690
+ delete this.#watchers[slot];
691
+ }), shareReplay(1));
692
+ }
693
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotsService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
694
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotsService }); }
695
+ }
696
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotsService, decorators: [{
697
+ type: Injectable
698
+ }], ctorParameters: () => [] });
699
+ class SlotDirective {
700
+ constructor() {
701
+ this.tpl = inject((TemplateRef));
702
+ }
703
+ set slot(slot) {
704
+ if (this.#slot !== slot) {
705
+ this.#slot = slot;
706
+ if (this.#slotDef) {
707
+ this.slotSvc.delTpl(this.#slotDef);
708
+ }
709
+ this.#slotDef = new SlotDef(slot, this.tpl);
710
+ this.slotSvc.addTpl(this.#slotDef);
711
+ }
712
+ }
713
+ get slot() {
714
+ return this.#slot;
715
+ }
716
+ #slot;
717
+ #slotDef;
718
+ ngOnDestroy() {
719
+ if (this.#slotDef) {
720
+ this.slotSvc.delTpl(this.#slotDef);
721
+ }
722
+ }
723
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
724
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.1.3", type: SlotDirective, ngImport: i0 }); }
725
+ }
726
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotDirective, decorators: [{
727
+ type: Directive
728
+ }] });
729
+ class SlotOutletDirective extends Destructible {
730
+ set slot(slot) {
731
+ if (this.#slot.value !== slot) {
732
+ this.#slot.next(slot);
733
+ }
734
+ }
735
+ get slot() {
736
+ return this.#slot.value;
737
+ }
738
+ #slot;
739
+ #watch;
740
+ #views;
741
+ constructor() {
742
+ super();
743
+ this.vcr = inject(ViewContainerRef);
744
+ this.injector = inject(Injector);
745
+ this.#slot = new BehaviorSubject(null);
746
+ this.#watch = this.#slot.pipe(switchMap(slot => {
747
+ if (slot) {
748
+ return this.slotSvc.watch(slot);
749
+ }
750
+ else {
751
+ return of([]);
752
+ }
753
+ }));
754
+ this.#views = [];
755
+ this.#onEntriesChanged = (entries) => {
756
+ const { remove, undecided } = this.#determineActions(entries);
757
+ for (const r of remove) {
758
+ r.dispose();
759
+ const idx = this.#views.indexOf(r);
760
+ if (idx >= 0) {
761
+ this.#views.splice(idx, 1);
762
+ }
763
+ }
764
+ this.#views.length = 0;
765
+ for (const [pos, entry] of undecided.entries()) {
766
+ if (entry.viewRef && !entry.viewRef.destroyed) {
767
+ const currentPos = this.vcr.indexOf(entry.viewRef);
768
+ if (currentPos !== pos) {
769
+ this.vcr.insert(entry.viewRef, pos);
770
+ }
771
+ }
772
+ else {
773
+ ;
774
+ entry.viewRef = this.vcr.createEmbeddedView(entry.tpl, null, {
775
+ index: pos,
776
+ injector: this.injector
777
+ });
778
+ entry.viewRef.markForCheck();
779
+ }
780
+ this.#views.push(entry);
781
+ }
782
+ };
783
+ this.d.any(this.#clearViews.bind(this));
784
+ }
785
+ ngOnInit() {
786
+ this.d.sub(this.#watch).subscribe(this.#onEntriesChanged);
787
+ }
788
+ #onEntriesChanged;
789
+ #determineActions(entries) {
790
+ const byId = {};
791
+ let remove = [];
792
+ const undecided = [];
793
+ for (const entry of entries) {
794
+ if (entry.id != null) {
795
+ if (!byId[entry.id]) {
796
+ byId[entry.id] = [entry];
797
+ }
798
+ else {
799
+ byId[entry.id].push(entry);
800
+ }
801
+ }
802
+ else {
803
+ undecided.push(entry);
804
+ }
805
+ }
806
+ for (const values of Object.values(byId)) {
807
+ remove = remove.concat(values.slice(0, -1));
808
+ undecided.push(values[values.length - 1]);
809
+ }
810
+ for (const current of this.#views) {
811
+ if (!undecided.includes(current)) {
812
+ remove.push(current);
813
+ }
814
+ }
815
+ return { remove, undecided };
816
+ }
817
+ #clearViews() {
818
+ this.vcr.clear();
819
+ this.#views = [];
820
+ }
821
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotOutletDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
822
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.1.3", type: SlotOutletDirective, usesInheritance: true, ngImport: i0 }); }
823
+ }
824
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotOutletDirective, decorators: [{
825
+ type: Directive
826
+ }], ctorParameters: () => [] });
827
+
742
828
  /**
743
829
  * Generated bundle index. Do not edit.
744
830
  */
745
831
 
746
- export { DockingContentComponent, DockingLayoutComponent, DockingPanelComponent, FastDOM, L9Cell, L9Range, L9State, NuDockingLayout, watchDimension, watchMedia };
832
+ export { DockingContentComponent, DockingLayoutComponent, DockingPanelComponent, L9Cell, L9Range, L9State, NuDockingLayout, SlotDef, SlotDirective, SlotOutletDirective, SlotsService, watchDimension, watchMedia };
747
833
  //# sourceMappingURL=ngutil-layout.mjs.map