@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.
- package/esm2022/docking/docking-layout.component.mjs +2 -83
- package/esm2022/docking/docking-panel.component.mjs +3 -4
- package/esm2022/index.mjs +2 -1
- package/esm2022/services/slots.service.mjs +240 -0
- package/esm2022/util/dimension-watcher.mjs +4 -3
- package/esm2022/util/index.mjs +1 -2
- package/fesm2022/ngutil-layout.mjs +243 -157
- package/fesm2022/ngutil-layout.mjs.map +1 -1
- package/index.d.ts +1 -0
- package/package.json +2 -2
- package/services/slots.service.d.ts +69 -0
- package/util/index.d.ts +0 -1
- package/esm2022/util/dom.mjs +0 -71
- package/util/dom.d.ts +0 -17
|
@@ -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
|
|
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,
|
|
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,
|
|
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"]}
|
package/esm2022/util/index.mjs
CHANGED
|
@@ -1,4 +1,3 @@
|
|
|
1
1
|
export { watchMedia } from "./media-watcher";
|
|
2
2
|
export { watchDimension } from "./dimension-watcher";
|
|
3
|
-
|
|
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,
|
|
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
|
|
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,
|
|
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,
|
|
832
|
+
export { DockingContentComponent, DockingLayoutComponent, DockingPanelComponent, L9Cell, L9Range, L9State, NuDockingLayout, SlotDef, SlotDirective, SlotOutletDirective, SlotsService, watchDimension, watchMedia };
|
|
747
833
|
//# sourceMappingURL=ngutil-layout.mjs.map
|