@ngutil/layout 0.0.3-dev.1 → 0.0.3-dev.10

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.
Files changed (49) hide show
  1. package/docking/docking-content.component.d.ts +5 -0
  2. package/docking/docking-layout.component.d.ts +22 -0
  3. package/docking/docking-panel.component.d.ts +40 -0
  4. package/docking/index.d.ts +13 -0
  5. package/esm2022/docking/docking-content.component.mjs +11 -0
  6. package/esm2022/docking/docking-layout.component.mjs +140 -0
  7. package/esm2022/docking/docking-panel.component.mjs +179 -0
  8. package/esm2022/docking/index.mjs +20 -0
  9. package/esm2022/index.mjs +5 -0
  10. package/esm2022/l9/index.mjs +3 -0
  11. package/esm2022/l9/range.mjs +109 -0
  12. package/esm2022/l9/state.mjs +29 -0
  13. package/esm2022/ngutil-layout.mjs +5 -0
  14. package/esm2022/services/slots.service.mjs +240 -0
  15. package/esm2022/util/dimension-watcher.mjs +85 -0
  16. package/esm2022/util/dimension.mjs +2 -0
  17. package/esm2022/util/index.mjs +3 -0
  18. package/esm2022/util/media-watcher.mjs +29 -0
  19. package/esm2022/util/rect.mjs +2 -0
  20. package/fesm2022/ngutil-layout.mjs +822 -0
  21. package/fesm2022/ngutil-layout.mjs.map +1 -0
  22. package/index.d.ts +4 -0
  23. package/l9/index.d.ts +2 -0
  24. package/l9/range.d.ts +28 -0
  25. package/l9/state.d.ts +14 -0
  26. package/package.json +18 -5
  27. package/services/slots.service.d.ts +69 -0
  28. package/util/dimension-watcher.d.ts +4 -0
  29. package/util/dimension.d.ts +5 -0
  30. package/util/index.d.ts +4 -0
  31. package/util/media-watcher.d.ts +5 -0
  32. package/util/rect.d.ts +5 -0
  33. package/.eslintrc.json +0 -25
  34. package/.storybook/main.ts +0 -16
  35. package/.storybook/preview.ts +0 -0
  36. package/.storybook/tsconfig.json +0 -16
  37. package/jest.config.ts +0 -22
  38. package/ng-package.json +0 -7
  39. package/project.json +0 -73
  40. package/src/index.ts +0 -1
  41. package/src/lib/layout/layout.component.scss +0 -0
  42. package/src/lib/layout/layout.component.spec.ts +0 -22
  43. package/src/lib/layout/layout.component.stories.ts +0 -24
  44. package/src/lib/layout/layout.component.ts +0 -11
  45. package/src/test-setup.ts +0 -9
  46. package/tsconfig.json +0 -32
  47. package/tsconfig.lib.json +0 -19
  48. package/tsconfig.lib.prod.json +0 -9
  49. package/tsconfig.spec.json +0 -11
@@ -0,0 +1,109 @@
1
+ /**
2
+ * -----------------------------------------------
3
+ * | TOP:LEFT | TOP:CENTER | TOP:RIGHT |
4
+ * -----------------------------------------------
5
+ * | MIDDLE:LEFT | MIDDLE:CENTER | MIDDLE:RIGH |
6
+ * -----------------------------------------------
7
+ * | BOTTOMN:LEFT | BOTTOM:CENTER | BOTTOM:RIGHT |
8
+ * -----------------------------------------------
9
+ */
10
+ const vertical = ["top", "middle", "bottom"];
11
+ const horizontal = ["left", "center", "right"];
12
+ export class L9Cell {
13
+ static coerce(value) {
14
+ const [v, h] = value.split(":");
15
+ if (vertical.includes(v) && horizontal.includes(h)) {
16
+ return new L9Cell(v, h);
17
+ }
18
+ throw new Error(`Invalid cell value: ${value}`);
19
+ }
20
+ constructor(v, h) {
21
+ this.v = v;
22
+ this.h = h;
23
+ }
24
+ }
25
+ export class L9Range {
26
+ static coerce(value) {
27
+ if (value instanceof L9Range) {
28
+ return value;
29
+ }
30
+ else {
31
+ return new L9Range(value);
32
+ }
33
+ }
34
+ constructor(range) {
35
+ this.cells = parse(range);
36
+ this.orient = this.#determineOrient();
37
+ }
38
+ isEq(other) {
39
+ if (other instanceof L9Range) {
40
+ const selfFirst = this.cells[0];
41
+ const otherFirst = other.cells[0];
42
+ if (selfFirst.h !== otherFirst.h || selfFirst.v !== otherFirst.v) {
43
+ return false;
44
+ }
45
+ const selfLast = this.cells[this.cells.length - 1];
46
+ const otherLast = other.cells[other.cells.length - 1];
47
+ if (selfLast.h === otherLast.h && selfLast.v === otherLast.v) {
48
+ return true;
49
+ }
50
+ }
51
+ return false;
52
+ }
53
+ #determineOrient() {
54
+ const { v: sv, h: sh } = this.cells[0];
55
+ const { v: ev, h: eh } = this.cells[this.cells.length - 1];
56
+ if (sv === ev) {
57
+ return "horizontal";
58
+ }
59
+ else if (sh === eh) {
60
+ return "vertical";
61
+ }
62
+ else {
63
+ return "rect";
64
+ }
65
+ }
66
+ }
67
+ function parse(value) {
68
+ const cells = [];
69
+ if (vertical.includes(value)) {
70
+ for (const h of horizontal) {
71
+ cells.push(new L9Cell(value, h));
72
+ }
73
+ }
74
+ else if (horizontal.includes(value)) {
75
+ for (const v of vertical) {
76
+ cells.push(new L9Cell(v, value));
77
+ }
78
+ }
79
+ else if (value.includes("-")) {
80
+ const [begin, end] = value.split("-");
81
+ const beginCells = parse(begin);
82
+ const endCells = parse(end);
83
+ if (beginCells.length > 1) {
84
+ throw new Error(`Currently not supported begin range value: ${begin}`);
85
+ }
86
+ if (endCells.length > 1) {
87
+ throw new Error(`Currently not supported end range value: ${end}`);
88
+ }
89
+ const { v: bv, h: bh } = beginCells[0];
90
+ const { v: ev, h: eh } = endCells[0];
91
+ const vstart = Math.min(vertical.indexOf(bv), vertical.indexOf(ev));
92
+ const vend = Math.max(vertical.indexOf(bv), vertical.indexOf(ev));
93
+ const hstart = Math.min(horizontal.indexOf(bh), horizontal.indexOf(eh));
94
+ const hend = Math.max(horizontal.indexOf(bh), horizontal.indexOf(eh));
95
+ for (let vi = vstart; vi <= vend; vi++) {
96
+ for (let hi = hstart; hi <= hend; hi++) {
97
+ cells.push(new L9Cell(vertical[vi], horizontal[hi]));
98
+ }
99
+ }
100
+ }
101
+ else if (value.includes(":")) {
102
+ cells.push(L9Cell.coerce(value));
103
+ }
104
+ if (cells.length === 0) {
105
+ throw Error(`Undefined l9cell: "${value}"`);
106
+ }
107
+ return cells;
108
+ }
109
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"range.js","sourceRoot":"","sources":["../../../../../packages/layout/src/l9/range.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * -----------------------------------------------\n * | TOP:LEFT     |  TOP:CENTER   | TOP:RIGHT    |\n * -----------------------------------------------\n * | MIDDLE:LEFT  | MIDDLE:CENTER | MIDDLE:RIGH  |\n * -----------------------------------------------\n * | BOTTOMN:LEFT | BOTTOM:CENTER | BOTTOM:RIGHT |\n * -----------------------------------------------\n */\n\nconst vertical = [\"top\", \"middle\", \"bottom\"] as const\nconst horizontal = [\"left\", \"center\", \"right\"] as const\n\nexport type L9Vertical = \"top\" | \"middle\" | \"bottom\"\nexport type L9Horizontal = \"left\" | \"center\" | \"right\"\nexport type L9CellName = `${L9Vertical}:${L9Horizontal}`\nexport type L9RangeName = L9Vertical | L9Horizontal | L9CellName | `${L9CellName}-${L9CellName}`\nexport type L9Orient = \"horizontal\" | \"vertical\" | \"rect\"\n\nexport class L9Cell {\n    static coerce(value: L9CellName) {\n        const [v, h] = value.split(\":\")\n\n        if (vertical.includes(v as any) && horizontal.includes(h as any)) {\n            return new L9Cell(v as any, h as any)\n        }\n\n        throw new Error(`Invalid cell value: ${value}`)\n    }\n\n    constructor(\n        public readonly v: L9Vertical,\n        public readonly h: L9Horizontal\n    ) {}\n}\n\nexport class L9Range {\n    static coerce(value: L9Range | L9RangeName) {\n        if (value instanceof L9Range) {\n            return value\n        } else {\n            return new L9Range(value)\n        }\n    }\n\n    readonly cells: Array<L9Cell>\n    readonly orient: L9Orient\n\n    constructor(range: L9RangeName) {\n        this.cells = parse(range)\n        this.orient = this.#determineOrient()\n    }\n\n    isEq(other: L9Range) {\n        if (other instanceof L9Range) {\n            const selfFirst = this.cells[0]\n            const otherFirst = other.cells[0]\n\n            if (selfFirst.h !== otherFirst.h || selfFirst.v !== otherFirst.v) {\n                return false\n            }\n\n            const selfLast = this.cells[this.cells.length - 1]\n            const otherLast = other.cells[other.cells.length - 1]\n\n            if (selfLast.h === otherLast.h && selfLast.v === otherLast.v) {\n                return true\n            }\n        }\n        return false\n    }\n\n    #determineOrient(): L9Orient {\n        const { v: sv, h: sh } = this.cells[0]\n        const { v: ev, h: eh } = this.cells[this.cells.length - 1]\n\n        if (sv === ev) {\n            return \"horizontal\"\n        } else if (sh === eh) {\n            return \"vertical\"\n        } else {\n            return \"rect\"\n        }\n    }\n}\n\nfunction parse(value: any): Array<L9Cell> {\n    const cells: Array<L9Cell> = []\n\n    if (vertical.includes(value)) {\n        for (const h of horizontal) {\n            cells.push(new L9Cell(value, h))\n        }\n    } else if (horizontal.includes(value)) {\n        for (const v of vertical) {\n            cells.push(new L9Cell(v, value))\n        }\n    } else if (value.includes(\"-\")) {\n        const [begin, end] = value.split(\"-\")\n        const beginCells = parse(begin)\n        const endCells = parse(end)\n\n        if (beginCells.length > 1) {\n            throw new Error(`Currently not supported begin range value: ${begin}`)\n        }\n\n        if (endCells.length > 1) {\n            throw new Error(`Currently not supported end range value: ${end}`)\n        }\n\n        const { v: bv, h: bh } = beginCells[0]\n        const { v: ev, h: eh } = endCells[0]\n\n        const vstart = Math.min(vertical.indexOf(bv as any), vertical.indexOf(ev as any))\n        const vend = Math.max(vertical.indexOf(bv as any), vertical.indexOf(ev as any))\n        const hstart = Math.min(horizontal.indexOf(bh as any), horizontal.indexOf(eh as any))\n        const hend = Math.max(horizontal.indexOf(bh as any), horizontal.indexOf(eh as any))\n\n        for (let vi = vstart; vi <= vend; vi++) {\n            for (let hi = hstart; hi <= hend; hi++) {\n                cells.push(new L9Cell(vertical[vi], horizontal[hi]))\n            }\n        }\n    } else if (value.includes(\":\")) {\n        cells.push(L9Cell.coerce(value))\n    }\n\n    if (cells.length === 0) {\n        throw Error(`Undefined l9cell: \"${value}\"`)\n    }\n\n    return cells\n}\n"]}
@@ -0,0 +1,29 @@
1
+ import { BehaviorSubject, map, shareReplay } from "rxjs";
2
+ import { L9Range } from "./range";
3
+ export class L9State {
4
+ #dims;
5
+ constructor(prefix) {
6
+ this.prefix = prefix;
7
+ this.#dims = new BehaviorSubject({});
8
+ this.style = this.#dims.pipe(map(dims => {
9
+ const res = {};
10
+ for (const [k, v] of Object.entries(dims)) {
11
+ if (v == null) {
12
+ continue;
13
+ }
14
+ const [vertical, horizontal] = k.split(":");
15
+ res[`--${this.prefix}-${vertical}-${horizontal}-w`] = v.width.toString();
16
+ res[`--${this.prefix}-${vertical}-${horizontal}-h`] = v.height.toString();
17
+ }
18
+ return res;
19
+ }), shareReplay(1));
20
+ }
21
+ update(range, dim) {
22
+ range = L9Range.coerce(range);
23
+ const dims = { ...this.#dims.value };
24
+ // for (const cell of range.horizontals) {
25
+ // dims[cell] = dim.width
26
+ // }
27
+ }
28
+ }
29
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhdGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9sYXlvdXQvc3JjL2w5L3N0YXRlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxlQUFlLEVBQUUsR0FBRyxFQUFjLFdBQVcsRUFBRSxNQUFNLE1BQU0sQ0FBQTtBQUdwRSxPQUFPLEVBQTRCLE9BQU8sRUFBMkIsTUFBTSxTQUFTLENBQUE7QUFPcEYsTUFBTSxPQUFPLE9BQU87SUFDUCxLQUFLLENBQWtDO0lBcUJoRCxZQUE0QixNQUFTO1FBQVQsV0FBTSxHQUFOLE1BQU0sQ0FBRztRQXJCNUIsVUFBSyxHQUFHLElBQUksZUFBZSxDQUFTLEVBQUUsQ0FBQyxDQUFBO1FBRXZDLFVBQUssR0FBK0IsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQ3hELEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRTtZQUNQLE1BQU0sR0FBRyxHQUFtQixFQUFFLENBQUE7WUFFOUIsS0FBSyxNQUFNLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxJQUFJLE1BQU0sQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQztnQkFDeEMsSUFBSSxDQUFDLElBQUksSUFBSSxFQUFFLENBQUM7b0JBQ1osU0FBUTtnQkFDWixDQUFDO2dCQUVELE1BQU0sQ0FBQyxRQUFRLEVBQUUsVUFBVSxDQUFDLEdBQUcsQ0FBQyxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQTtnQkFDM0MsR0FBRyxDQUFDLEtBQUssSUFBSSxDQUFDLE1BQU0sSUFBSSxRQUFRLElBQUksVUFBVSxJQUFxQixDQUFDLEdBQUcsQ0FBQyxDQUFDLEtBQUssQ0FBQyxRQUFRLEVBQUUsQ0FBQTtnQkFDekYsR0FBRyxDQUFDLEtBQUssSUFBSSxDQUFDLE1BQU0sSUFBSSxRQUFRLElBQUksVUFBVSxJQUFxQixDQUFDLEdBQUcsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxRQUFRLEVBQUUsQ0FBQTtZQUM5RixDQUFDO1lBRUQsT0FBTyxHQUFHLENBQUE7UUFDZCxDQUFDLENBQUMsRUFDRixXQUFXLENBQUMsQ0FBQyxDQUFDLENBQ2pCLENBQUE7SUFFdUMsQ0FBQztJQUV6QyxNQUFNLENBQUMsS0FBNEIsRUFBRSxHQUFjO1FBQy9DLEtBQUssR0FBRyxPQUFPLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFBO1FBQzdCLE1BQU0sSUFBSSxHQUFHLEVBQUUsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFBO1FBQ3BDLDBDQUEwQztRQUMxQyw2QkFBNkI7UUFDN0IsSUFBSTtJQUNSLENBQUM7Q0FDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEJlaGF2aW9yU3ViamVjdCwgbWFwLCBPYnNlcnZhYmxlLCBzaGFyZVJlcGxheSB9IGZyb20gXCJyeGpzXCJcblxuaW1wb3J0IHsgRGltZW5zaW9uIH0gZnJvbSBcIi4uL3V0aWxcIlxuaW1wb3J0IHsgTDlDZWxsTmFtZSwgTDlIb3Jpem9udGFsLCBMOVJhbmdlLCBMOVJhbmdlTmFtZSwgTDlWZXJ0aWNhbCB9IGZyb20gXCIuL3JhbmdlXCJcblxuZXhwb3J0IHR5cGUgTDlTdGF0ZVZhcjxUIGV4dGVuZHMgc3RyaW5nPiA9IGAtLSR7VH0tJHtMOVZlcnRpY2FsfS0ke0w5SG9yaXpvbnRhbH0tJHtcIndcIiB8IFwiaFwifWBcbmV4cG9ydCB0eXBlIEw5U3R5bGVWYXJzPFQgZXh0ZW5kcyBzdHJpbmc+ID0geyBba2V5IGluIEw5U3RhdGVWYXI8VD5dPzogc3RyaW5nIH1cblxudHlwZSBMOURpbXMgPSB7IFtrZXkgaW4gTDlDZWxsTmFtZV0/OiBEaW1lbnNpb24gfVxuXG5leHBvcnQgY2xhc3MgTDlTdGF0ZTxUIGV4dGVuZHMgc3RyaW5nPiB7XG4gICAgcmVhZG9ubHkgI2RpbXMgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PEw5RGltcz4oe30pXG5cbiAgICByZWFkb25seSBzdHlsZTogT2JzZXJ2YWJsZTxMOVN0eWxlVmFyczxUPj4gPSB0aGlzLiNkaW1zLnBpcGUoXG4gICAgICAgIG1hcChkaW1zID0+IHtcbiAgICAgICAgICAgIGNvbnN0IHJlczogTDlTdHlsZVZhcnM8VD4gPSB7fVxuXG4gICAgICAgICAgICBmb3IgKGNvbnN0IFtrLCB2XSBvZiBPYmplY3QuZW50cmllcyhkaW1zKSkge1xuICAgICAgICAgICAgICAgIGlmICh2ID09IG51bGwpIHtcbiAgICAgICAgICAgICAgICAgICAgY29udGludWVcbiAgICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgICBjb25zdCBbdmVydGljYWwsIGhvcml6b250YWxdID0gay5zcGxpdChcIjpcIilcbiAgICAgICAgICAgICAgICByZXNbYC0tJHt0aGlzLnByZWZpeH0tJHt2ZXJ0aWNhbH0tJHtob3Jpem9udGFsfS13YCBhcyBMOVN0YXRlVmFyPFQ+XSA9IHYud2lkdGgudG9TdHJpbmcoKVxuICAgICAgICAgICAgICAgIHJlc1tgLS0ke3RoaXMucHJlZml4fS0ke3ZlcnRpY2FsfS0ke2hvcml6b250YWx9LWhgIGFzIEw5U3RhdGVWYXI8VD5dID0gdi5oZWlnaHQudG9TdHJpbmcoKVxuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICByZXR1cm4gcmVzXG4gICAgICAgIH0pLFxuICAgICAgICBzaGFyZVJlcGxheSgxKVxuICAgIClcblxuICAgIGNvbnN0cnVjdG9yKHB1YmxpYyByZWFkb25seSBwcmVmaXg6IFQpIHt9XG5cbiAgICB1cGRhdGUocmFuZ2U6IEw5UmFuZ2UgfCBMOVJhbmdlTmFtZSwgZGltOiBEaW1lbnNpb24pIHtcbiAgICAgICAgcmFuZ2UgPSBMOVJhbmdlLmNvZXJjZShyYW5nZSlcbiAgICAgICAgY29uc3QgZGltcyA9IHsgLi4udGhpcy4jZGltcy52YWx1ZSB9XG4gICAgICAgIC8vIGZvciAoY29uc3QgY2VsbCBvZiByYW5nZS5ob3Jpem9udGFscykge1xuICAgICAgICAvLyAgICAgZGltc1tjZWxsXSA9IGRpbS53aWR0aFxuICAgICAgICAvLyB9XG4gICAgfVxufVxuIl19
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './index';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmd1dGlsLWxheW91dC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL2xheW91dC9zcmMvbmd1dGlsLWxheW91dC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2luZGV4JztcbiJdfQ==
@@ -0,0 +1,240 @@
1
+ import { Directive, inject, Injectable, Injector, TemplateRef, ViewContainerRef } from "@angular/core";
2
+ import { BehaviorSubject, distinctUntilChanged, finalize, map, of, scan, shareReplay, Subject, switchMap, tap } from "rxjs";
3
+ import { Destructible } from "@ngutil/common";
4
+ import * as i0 from "@angular/core";
5
+ const SLOT_REGEX = /^([^:\s]+)(?::(\d+))?(?:\s+as\s+(.*?))?$/i;
6
+ export class SlotDef {
7
+ constructor(slot, tpl) {
8
+ this.tpl = tpl;
9
+ const match = slot.match(SLOT_REGEX);
10
+ if (!match) {
11
+ console.warn(`Invalid slot definition: ${slot}`);
12
+ }
13
+ else {
14
+ this.slot = match[1];
15
+ this.order = match[2] != null ? Number(match[2]) : Infinity;
16
+ this.id = match[3];
17
+ }
18
+ }
19
+ dispose() {
20
+ this.viewRef?.destroy();
21
+ this.viewRef = undefined;
22
+ }
23
+ }
24
+ /**
25
+ * @Directive({selector: "ng-template[xyzSlot]", inputs: [{name: "slot", alias: "xyzSlot"}]})
26
+ * class XYZSlotDirective extends SlotDirective<XYZComponentSlots> { }
27
+ *
28
+ * @Directive({selector: "ng-template[xyzSlotOutlet]", inputs: [{name: "slot", alias: "xyzSlotOutlet"}]})
29
+ * class XYZSlotOutletDirective extends SlotOutletDirective<XYZComponentSlots> { }
30
+ *
31
+ *
32
+ * @Component({provides: [SlotsService]})
33
+ * class XYZComponent {
34
+ * slots: inject(SlotsService<XYZComponentSlots>)
35
+ * }
36
+ *
37
+ *
38
+ */
39
+ export class SlotsService extends Destructible {
40
+ #events = new Subject();
41
+ #entries = this.#events.pipe(scan((entries, event) => {
42
+ if (event.type === "add") {
43
+ const index = entries.findIndex(value => value === event.def);
44
+ if (index > -1) {
45
+ entries[index] = event.def;
46
+ }
47
+ else {
48
+ entries.push(event.def);
49
+ }
50
+ }
51
+ else if (event.type === "del") {
52
+ const index = entries.findIndex(value => value === event.def);
53
+ if (index > -1) {
54
+ entries.splice(index, 1);
55
+ }
56
+ }
57
+ return entries;
58
+ }, []), tap(entries => {
59
+ entries.sort((a, b) => {
60
+ if (a.slot === b.slot) {
61
+ return a.order - b.order;
62
+ }
63
+ else {
64
+ return a.slot.localeCompare(b.slot);
65
+ }
66
+ });
67
+ }), shareReplay(1));
68
+ constructor() {
69
+ super();
70
+ // XXX: need to collect entries from the beginning
71
+ this.d.sub(this.#entries).subscribe();
72
+ }
73
+ addTpl(def) {
74
+ this.#events.next({ type: "add", def });
75
+ }
76
+ delTpl(def) {
77
+ this.#events.next({ type: "del", def });
78
+ }
79
+ #watchers = {};
80
+ watch(slot) {
81
+ const existing = this.#watchers[slot];
82
+ if (existing == null) {
83
+ return (this.#watchers[slot] = this.#watch(slot));
84
+ }
85
+ else {
86
+ return existing;
87
+ }
88
+ }
89
+ #watch(slot) {
90
+ return this.#entries.pipe(map(entries => entries.filter(entry => entry.slot === slot)), distinctUntilChanged((prev, curr) => {
91
+ if (prev.length === curr.length) {
92
+ for (let i = 0; i < prev.length; i++) {
93
+ if (prev[i] !== curr[i]) {
94
+ return false;
95
+ }
96
+ }
97
+ return true;
98
+ }
99
+ else {
100
+ return false;
101
+ }
102
+ }), finalize(() => {
103
+ delete this.#watchers[slot];
104
+ }), shareReplay(1));
105
+ }
106
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotsService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
107
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotsService }); }
108
+ }
109
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotsService, decorators: [{
110
+ type: Injectable
111
+ }], ctorParameters: () => [] });
112
+ export class SlotDirective {
113
+ constructor() {
114
+ this.tpl = inject((TemplateRef));
115
+ }
116
+ set slot(slot) {
117
+ if (this.#slot !== slot) {
118
+ this.#slot = slot;
119
+ if (this.#slotDef) {
120
+ this.slotSvc.delTpl(this.#slotDef);
121
+ }
122
+ this.#slotDef = new SlotDef(slot, this.tpl);
123
+ this.slotSvc.addTpl(this.#slotDef);
124
+ }
125
+ }
126
+ get slot() {
127
+ return this.#slot;
128
+ }
129
+ #slot;
130
+ #slotDef;
131
+ ngOnDestroy() {
132
+ if (this.#slotDef) {
133
+ this.slotSvc.delTpl(this.#slotDef);
134
+ }
135
+ }
136
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
137
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.1.3", type: SlotDirective, ngImport: i0 }); }
138
+ }
139
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotDirective, decorators: [{
140
+ type: Directive
141
+ }] });
142
+ export class SlotOutletDirective extends Destructible {
143
+ set slot(slot) {
144
+ if (this.#slot.value !== slot) {
145
+ this.#slot.next(slot);
146
+ }
147
+ }
148
+ get slot() {
149
+ return this.#slot.value;
150
+ }
151
+ #slot;
152
+ #watch;
153
+ #views;
154
+ constructor() {
155
+ super();
156
+ this.vcr = inject(ViewContainerRef);
157
+ this.injector = inject(Injector);
158
+ this.#slot = new BehaviorSubject(null);
159
+ this.#watch = this.#slot.pipe(switchMap(slot => {
160
+ if (slot) {
161
+ return this.slotSvc.watch(slot);
162
+ }
163
+ else {
164
+ return of([]);
165
+ }
166
+ }));
167
+ this.#views = [];
168
+ this.#onEntriesChanged = (entries) => {
169
+ const { remove, undecided } = this.#determineActions(entries);
170
+ for (const r of remove) {
171
+ r.dispose();
172
+ const idx = this.#views.indexOf(r);
173
+ if (idx >= 0) {
174
+ this.#views.splice(idx, 1);
175
+ }
176
+ }
177
+ this.#views.length = 0;
178
+ for (const [pos, entry] of undecided.entries()) {
179
+ if (entry.viewRef && !entry.viewRef.destroyed) {
180
+ const currentPos = this.vcr.indexOf(entry.viewRef);
181
+ if (currentPos !== pos) {
182
+ this.vcr.insert(entry.viewRef, pos);
183
+ }
184
+ }
185
+ else {
186
+ ;
187
+ entry.viewRef = this.vcr.createEmbeddedView(entry.tpl, null, {
188
+ index: pos,
189
+ injector: this.injector
190
+ });
191
+ entry.viewRef.markForCheck();
192
+ }
193
+ this.#views.push(entry);
194
+ }
195
+ };
196
+ this.d.any(this.#clearViews.bind(this));
197
+ }
198
+ ngOnInit() {
199
+ this.d.sub(this.#watch).subscribe(this.#onEntriesChanged);
200
+ }
201
+ #onEntriesChanged;
202
+ #determineActions(entries) {
203
+ const byId = {};
204
+ let remove = [];
205
+ const undecided = [];
206
+ for (const entry of entries) {
207
+ if (entry.id != null) {
208
+ if (!byId[entry.id]) {
209
+ byId[entry.id] = [entry];
210
+ }
211
+ else {
212
+ byId[entry.id].push(entry);
213
+ }
214
+ }
215
+ else {
216
+ undecided.push(entry);
217
+ }
218
+ }
219
+ for (const values of Object.values(byId)) {
220
+ remove = remove.concat(values.slice(0, -1));
221
+ undecided.push(values[values.length - 1]);
222
+ }
223
+ for (const current of this.#views) {
224
+ if (!undecided.includes(current)) {
225
+ remove.push(current);
226
+ }
227
+ }
228
+ return { remove, undecided };
229
+ }
230
+ #clearViews() {
231
+ this.vcr.clear();
232
+ this.#views = [];
233
+ }
234
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotOutletDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
235
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.1.3", type: SlotOutletDirective, usesInheritance: true, ngImport: i0 }); }
236
+ }
237
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.3", ngImport: i0, type: SlotOutletDirective, decorators: [{
238
+ type: Directive
239
+ }], ctorParameters: () => [] });
240
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slots.service.js","sourceRoot":"","sources":["../../../../../packages/layout/src/services/slots.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Directive,\n    EmbeddedViewRef,\n    inject,\n    Injectable,\n    Injector,\n    OnDestroy,\n    OnInit,\n    TemplateRef,\n    ViewContainerRef\n} from \"@angular/core\"\n\nimport {\n    BehaviorSubject,\n    distinctUntilChanged,\n    finalize,\n    map,\n    Observable,\n    of,\n    scan,\n    shareReplay,\n    Subject,\n    switchMap,\n    tap\n} from \"rxjs\"\n\nimport { Destructible, IDisposable } from \"@ngutil/common\"\n\ntype SlotOrder = `:${number}`\ntype SlotId = ` as ${string}`\nexport type Slot<T extends string> = `${T}${SlotOrder | \"\"}${SlotId | \"\"}`\nexport type SlotTpl = TemplateRef<any>\nexport type SlotViewRef = EmbeddedViewRef<any>\n\nconst SLOT_REGEX = /^([^:\\s]+)(?::(\\d+))?(?:\\s+as\\s+(.*?))?$/i\n\nexport class SlotDef<T extends string> implements IDisposable {\n    readonly slot!: T\n    readonly order!: number\n    readonly id?: string\n    readonly viewRef?: SlotViewRef\n\n    constructor(\n        slot: Slot<T>,\n        public readonly tpl: SlotTpl\n    ) {\n        const match = slot.match(SLOT_REGEX)\n        if (!match) {\n            console.warn(`Invalid slot definition: ${slot}`)\n        } else {\n            this.slot = match[1] as any\n            this.order = match[2] != null ? Number(match[2]) : Infinity\n            this.id = match[3] as string\n        }\n    }\n\n    dispose(): void {\n        this.viewRef?.destroy()\n        ;(this as { viewRef?: SlotViewRef }).viewRef = undefined\n    }\n}\n\nexport interface SlotEvent<T extends string> {\n    type: \"add\" | \"del\"\n    def: SlotDef<T>\n}\n\n/**\n * @Directive({selector: \"ng-template[xyzSlot]\", inputs: [{name: \"slot\", alias: \"xyzSlot\"}]})\n * class XYZSlotDirective extends SlotDirective<XYZComponentSlots> { }\n *\n * @Directive({selector: \"ng-template[xyzSlotOutlet]\", inputs: [{name: \"slot\", alias: \"xyzSlotOutlet\"}]})\n * class XYZSlotOutletDirective extends SlotOutletDirective<XYZComponentSlots> { }\n *\n *\n * @Component({provides: [SlotsService]})\n * class XYZComponent {\n *      slots: inject(SlotsService<XYZComponentSlots>)\n * }\n *\n *\n */\n@Injectable()\nexport class SlotsService<T extends string = any> extends Destructible {\n    #events = new Subject<SlotEvent<T>>()\n\n    #entries = this.#events.pipe(\n        scan((entries, event) => {\n            if (event.type === \"add\") {\n                const index = entries.findIndex(value => value === event.def)\n                if (index > -1) {\n                    entries[index] = event.def\n                } else {\n                    entries.push(event.def)\n                }\n            } else if (event.type === \"del\") {\n                const index = entries.findIndex(value => value === event.def)\n                if (index > -1) {\n                    entries.splice(index, 1)\n                }\n            }\n            return entries\n        }, [] as SlotDef<T>[]),\n        tap(entries => {\n            entries.sort((a, b) => {\n                if (a.slot === b.slot) {\n                    return a.order - b.order\n                } else {\n                    return a.slot.localeCompare(b.slot)\n                }\n            })\n        }),\n        shareReplay(1)\n    )\n\n    constructor() {\n        super()\n        // XXX: need to collect entries from the beginning\n        this.d.sub(this.#entries).subscribe()\n    }\n\n    addTpl(def: SlotDef<T>) {\n        this.#events.next({ type: \"add\", def })\n    }\n\n    delTpl(def: SlotDef<T>) {\n        this.#events.next({ type: \"del\", def })\n    }\n\n    #watchers: { [key in T]: Observable<Array<SlotDef<T>>> } = {} as any\n\n    watch(slot: T) {\n        const existing = this.#watchers[slot]\n        if (existing == null) {\n            return (this.#watchers[slot] = this.#watch(slot))\n        } else {\n            return existing\n        }\n    }\n\n    #watch(slot: T) {\n        return this.#entries.pipe(\n            map(entries => entries.filter(entry => entry.slot === slot)),\n            distinctUntilChanged((prev, curr) => {\n                if (prev.length === curr.length) {\n                    for (let i = 0; i < prev.length; i++) {\n                        if (prev[i] !== curr[i]) {\n                            return false\n                        }\n                    }\n                    return true\n                } else {\n                    return false\n                }\n            }),\n            finalize(() => {\n                delete this.#watchers[slot]\n            }),\n            shareReplay(1)\n        )\n    }\n}\n\n@Directive()\nexport abstract class SlotDirective<T extends string, C = any> implements OnDestroy {\n    protected abstract readonly slotSvc: SlotsService<T>\n    protected readonly tpl = inject(TemplateRef<C>)\n\n    set slot(slot: Slot<T>) {\n        if (this.#slot !== slot) {\n            this.#slot = slot\n\n            if (this.#slotDef) {\n                this.slotSvc.delTpl(this.#slotDef)\n            }\n\n            this.#slotDef = new SlotDef<T>(slot, this.tpl)\n            this.slotSvc.addTpl(this.#slotDef)\n        }\n    }\n    get slot() {\n        return this.#slot\n    }\n    #slot!: Slot<T>\n    #slotDef?: SlotDef<T>\n\n    ngOnDestroy(): void {\n        if (this.#slotDef) {\n            this.slotSvc.delTpl(this.#slotDef)\n        }\n    }\n}\n\n@Directive()\nexport abstract class SlotOutletDirective<T extends string> extends Destructible implements OnInit {\n    protected abstract readonly slotSvc: SlotsService<T>\n    protected readonly vcr: ViewContainerRef = inject(ViewContainerRef)\n    protected readonly injector: Injector = inject(Injector)\n\n    set slot(slot: T | null) {\n        if (this.#slot.value !== slot) {\n            this.#slot.next(slot)\n        }\n    }\n    get slot() {\n        return this.#slot.value\n    }\n    #slot = new BehaviorSubject<T | null>(null)\n\n    #watch = this.#slot.pipe(\n        switchMap(slot => {\n            if (slot) {\n                return this.slotSvc.watch(slot)\n            } else {\n                return of([])\n            }\n        })\n    )\n\n    #views: Array<SlotDef<any>> = []\n\n    constructor() {\n        super()\n        this.d.any(this.#clearViews.bind(this))\n    }\n\n    ngOnInit(): void {\n        this.d.sub(this.#watch).subscribe(this.#onEntriesChanged)\n    }\n\n    #onEntriesChanged = (entries: Array<SlotDef<T>>) => {\n        const { remove, undecided } = this.#determineActions(entries)\n\n        for (const r of remove) {\n            r.dispose()\n            const idx = this.#views.indexOf(r)\n            if (idx >= 0) {\n                this.#views.splice(idx, 1)\n            }\n        }\n\n        this.#views.length = 0\n        for (const [pos, entry] of undecided.entries()) {\n            if (entry.viewRef && !entry.viewRef.destroyed) {\n                const currentPos = this.vcr.indexOf(entry.viewRef)\n                if (currentPos !== pos) {\n                    this.vcr.insert(entry.viewRef, pos)\n                }\n            } else {\n                ;(entry as { viewRef: SlotViewRef }).viewRef = this.vcr.createEmbeddedView(entry.tpl, null, {\n                    index: pos,\n                    injector: this.injector\n                })\n                entry.viewRef!.markForCheck()\n            }\n            this.#views.push(entry)\n        }\n    }\n\n    #determineActions(entries: Array<SlotDef<T>>) {\n        const byId: { [key: string]: Array<SlotDef<T>> } = {}\n        let remove: Array<SlotDef<T>> = []\n        const undecided: Array<SlotDef<T>> = []\n\n        for (const entry of entries) {\n            if (entry.id != null) {\n                if (!byId[entry.id]) {\n                    byId[entry.id] = [entry]\n                } else {\n                    byId[entry.id].push(entry)\n                }\n            } else {\n                undecided.push(entry)\n            }\n        }\n\n        for (const values of Object.values(byId)) {\n            remove = remove.concat(values.slice(0, -1))\n            undecided.push(values[values.length - 1])\n        }\n\n        for (const current of this.#views) {\n            if (!undecided.includes(current)) {\n                remove.push(current)\n            }\n        }\n\n        return { remove, undecided }\n    }\n\n    #clearViews() {\n        this.vcr.clear()\n        this.#views = []\n    }\n}\n"]}
@@ -0,0 +1,85 @@
1
+ import { inject, NgZone } from "@angular/core";
2
+ import { distinctUntilChanged, Observable, shareReplay } from "rxjs";
3
+ import { NumberWithUnit } from "@ngutil/common";
4
+ const RESIZE_WATCHES = new Map();
5
+ const SCROLL_WATCHES = new Map();
6
+ export function watchDimension(el, box = "border-box") {
7
+ const zone = inject(NgZone);
8
+ return box === "scroll-box" ? _watchScroll(zone, el) : _watchResize(zone, el, box);
9
+ }
10
+ function _watchResize(zone, el, box) {
11
+ return _watch(zone, el, RESIZE_WATCHES, () => _createResizeWatcher(zone, el, box));
12
+ }
13
+ function _watchScroll(zone, el) {
14
+ return _watch(zone, el, SCROLL_WATCHES, () => _createScollWatcher(zone, el));
15
+ }
16
+ function _watch(zone, el, watches, factory) {
17
+ const existing = watches.get(el);
18
+ if (existing == null) {
19
+ const watcher = factory();
20
+ watches.set(el, watcher);
21
+ return watcher;
22
+ }
23
+ return existing;
24
+ }
25
+ function _createResizeWatcher(zone, el, box) {
26
+ if (box !== "border-box") {
27
+ throw new Error(`Currently not implemented box mode: ${box}`);
28
+ }
29
+ return zone.runOutsideAngular(() => new Observable((sub) => {
30
+ const observer = new ResizeObserver(entries => {
31
+ for (const entry of entries) {
32
+ if (entry.borderBoxSize) {
33
+ sub.next({
34
+ width: _number(entry.borderBoxSize[0].inlineSize),
35
+ height: _number(entry.borderBoxSize[0].blockSize)
36
+ });
37
+ }
38
+ else {
39
+ sub.next({
40
+ width: _number(el.offsetWidth),
41
+ height: _number(el.offsetHeight)
42
+ });
43
+ }
44
+ }
45
+ });
46
+ observer.observe(el, { box: box });
47
+ return () => {
48
+ observer.disconnect();
49
+ RESIZE_WATCHES.delete(el);
50
+ };
51
+ }).pipe(distinctUntilChanged((p, c) => p && c && p.width === c.width && p.height === c.height), shareReplay(1)));
52
+ }
53
+ function _createScollWatcher(zone, el) {
54
+ return zone.runOutsideAngular(() => new Observable((sub) => {
55
+ let lastSw = NaN;
56
+ let lastSh = NaN;
57
+ const emit = () => {
58
+ const sw = el.scrollWidth;
59
+ const sh = el.scrollHeight;
60
+ if (lastSw !== sw || lastSh !== sh) {
61
+ lastSw = sw;
62
+ lastSh = sh;
63
+ sub.next({ width: _number(lastSw), height: _number(lastSh) });
64
+ }
65
+ };
66
+ const dimSum = _watchResize(zone, el, "border-box").subscribe(emit);
67
+ const mutation = new MutationObserver(emit);
68
+ mutation.observe(el, {
69
+ subtree: true,
70
+ childList: true,
71
+ attributes: true,
72
+ characterData: true
73
+ });
74
+ emit();
75
+ return () => {
76
+ dimSum.unsubscribe();
77
+ mutation.disconnect();
78
+ SCROLL_WATCHES.delete(el);
79
+ };
80
+ }).pipe(shareReplay(1)));
81
+ }
82
+ function _number(val) {
83
+ return new NumberWithUnit(val, "px");
84
+ }
85
+ //# 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 = NaN\n            let lastSh: number = NaN\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            emit()\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, \"px\")\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGltZW5zaW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvbGF5b3V0L3NyYy91dGlsL2RpbWVuc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTnVtYmVyV2l0aFVuaXQgfSBmcm9tIFwiQG5ndXRpbC9jb21tb25cIlxuXG5leHBvcnQgaW50ZXJmYWNlIERpbWVuc2lvbiB7XG4gICAgd2lkdGg6IE51bWJlcldpdGhVbml0XG4gICAgaGVpZ2h0OiBOdW1iZXJXaXRoVW5pdFxufVxuIl19
@@ -0,0 +1,3 @@
1
+ export { watchMedia } from "./media-watcher";
2
+ export { watchDimension } from "./dimension-watcher";
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9sYXlvdXQvc3JjL3V0aWwvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGlCQUFpQixDQUFBO0FBQzVDLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IERpbWVuc2lvbiB9IGZyb20gXCIuL2RpbWVuc2lvblwiXG5leHBvcnQgeyBSZWN0IH0gZnJvbSBcIi4vcmVjdFwiXG5leHBvcnQgeyB3YXRjaE1lZGlhIH0gZnJvbSBcIi4vbWVkaWEtd2F0Y2hlclwiXG5leHBvcnQgeyB3YXRjaERpbWVuc2lvbiB9IGZyb20gXCIuL2RpbWVuc2lvbi13YXRjaGVyXCJcbiJdfQ==
@@ -0,0 +1,29 @@
1
+ import { inject, NgZone } from "@angular/core";
2
+ import { distinctUntilChanged, Observable, shareReplay } from "rxjs";
3
+ const WATCHES = {};
4
+ /**
5
+ * watchMedia("(display-mode: standalone)")
6
+ */
7
+ export function watchMedia(expr) {
8
+ const existing = WATCHES[expr];
9
+ if (existing == null) {
10
+ return (WATCHES[expr] = _createWatcher(expr));
11
+ }
12
+ return existing;
13
+ }
14
+ function _createWatcher(expr) {
15
+ const zone = inject(NgZone);
16
+ return zone.runOutsideAngular(() => new Observable((sub) => {
17
+ const query = window.matchMedia(expr);
18
+ const listener = (event) => {
19
+ sub.next(event.matches);
20
+ };
21
+ query.addEventListener("change", listener);
22
+ sub.next(query.matches);
23
+ return () => {
24
+ query.removeEventListener("change", listener);
25
+ delete WATCHES[expr];
26
+ };
27
+ }).pipe(distinctUntilChanged(), shareReplay(1)));
28
+ }
29
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVkaWEtd2F0Y2hlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2xheW91dC9zcmMvdXRpbC9tZWRpYS13YXRjaGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFBO0FBRTlDLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxVQUFVLEVBQUUsV0FBVyxFQUFjLE1BQU0sTUFBTSxDQUFBO0FBRWhGLE1BQU0sT0FBTyxHQUEyQyxFQUFFLENBQUE7QUFFMUQ7O0dBRUc7QUFDSCxNQUFNLFVBQVUsVUFBVSxDQUFDLElBQVk7SUFDbkMsTUFBTSxRQUFRLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFBO0lBQzlCLElBQUksUUFBUSxJQUFJLElBQUksRUFBRSxDQUFDO1FBQ25CLE9BQU8sQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEdBQUcsY0FBYyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUE7SUFDakQsQ0FBQztJQUNELE9BQU8sUUFBUSxDQUFBO0FBQ25CLENBQUM7QUFFRCxTQUFTLGNBQWMsQ0FBQyxJQUFZO0lBQ2hDLE1BQU0sSUFBSSxHQUFHLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQTtJQUMzQixPQUFPLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxHQUFHLEVBQUUsQ0FDL0IsSUFBSSxVQUFVLENBQUMsQ0FBQyxHQUF3QixFQUFFLEVBQUU7UUFDeEMsTUFBTSxLQUFLLEdBQUcsTUFBTSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsQ0FBQTtRQUNyQyxNQUFNLFFBQVEsR0FBRyxDQUFDLEtBQTBCLEVBQUUsRUFBRTtZQUM1QyxHQUFHLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQTtRQUMzQixDQUFDLENBQUE7UUFDRCxLQUFLLENBQUMsZ0JBQWdCLENBQUMsUUFBUSxFQUFFLFFBQVEsQ0FBQyxDQUFBO1FBQzFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFBO1FBQ3ZCLE9BQU8sR0FBRyxFQUFFO1lBQ1IsS0FBSyxDQUFDLG1CQUFtQixDQUFDLFFBQVEsRUFBRSxRQUFRLENBQUMsQ0FBQTtZQUM3QyxPQUFPLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQTtRQUN4QixDQUFDLENBQUE7SUFDTCxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsb0JBQW9CLEVBQUUsRUFBRSxXQUFXLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FDbEQsQ0FBQTtBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBpbmplY3QsIE5nWm9uZSB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCJcblxuaW1wb3J0IHsgZGlzdGluY3RVbnRpbENoYW5nZWQsIE9ic2VydmFibGUsIHNoYXJlUmVwbGF5LCBTdWJzY3JpYmVyIH0gZnJvbSBcInJ4anNcIlxuXG5jb25zdCBXQVRDSEVTOiB7IFtrZXk6IHN0cmluZ106IE9ic2VydmFibGU8Ym9vbGVhbj4gfSA9IHt9XG5cbi8qKlxuICogd2F0Y2hNZWRpYShcIihkaXNwbGF5LW1vZGU6IHN0YW5kYWxvbmUpXCIpXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiB3YXRjaE1lZGlhKGV4cHI6IHN0cmluZyk6IE9ic2VydmFibGU8Ym9vbGVhbj4ge1xuICAgIGNvbnN0IGV4aXN0aW5nID0gV0FUQ0hFU1tleHByXVxuICAgIGlmIChleGlzdGluZyA9PSBudWxsKSB7XG4gICAgICAgIHJldHVybiAoV0FUQ0hFU1tleHByXSA9IF9jcmVhdGVXYXRjaGVyKGV4cHIpKVxuICAgIH1cbiAgICByZXR1cm4gZXhpc3Rpbmdcbn1cblxuZnVuY3Rpb24gX2NyZWF0ZVdhdGNoZXIoZXhwcjogc3RyaW5nKTogT2JzZXJ2YWJsZTxib29sZWFuPiB7XG4gICAgY29uc3Qgem9uZSA9IGluamVjdChOZ1pvbmUpXG4gICAgcmV0dXJuIHpvbmUucnVuT3V0c2lkZUFuZ3VsYXIoKCkgPT5cbiAgICAgICAgbmV3IE9ic2VydmFibGUoKHN1YjogU3Vic2NyaWJlcjxib29sZWFuPikgPT4ge1xuICAgICAgICAgICAgY29uc3QgcXVlcnkgPSB3aW5kb3cubWF0Y2hNZWRpYShleHByKVxuICAgICAgICAgICAgY29uc3QgbGlzdGVuZXIgPSAoZXZlbnQ6IE1lZGlhUXVlcnlMaXN0RXZlbnQpID0+IHtcbiAgICAgICAgICAgICAgICBzdWIubmV4dChldmVudC5tYXRjaGVzKVxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgcXVlcnkuYWRkRXZlbnRMaXN0ZW5lcihcImNoYW5nZVwiLCBsaXN0ZW5lcilcbiAgICAgICAgICAgIHN1Yi5uZXh0KHF1ZXJ5Lm1hdGNoZXMpXG4gICAgICAgICAgICByZXR1cm4gKCkgPT4ge1xuICAgICAgICAgICAgICAgIHF1ZXJ5LnJlbW92ZUV2ZW50TGlzdGVuZXIoXCJjaGFuZ2VcIiwgbGlzdGVuZXIpXG4gICAgICAgICAgICAgICAgZGVsZXRlIFdBVENIRVNbZXhwcl1cbiAgICAgICAgICAgIH1cbiAgICAgICAgfSkucGlwZShkaXN0aW5jdFVudGlsQ2hhbmdlZCgpLCBzaGFyZVJlcGxheSgxKSlcbiAgICApXG59XG4iXX0=
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2xheW91dC9zcmMvdXRpbC9yZWN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaW1lbnNpb24gfSBmcm9tIFwiLi9kaW1lbnNpb25cIlxuXG5leHBvcnQgaW50ZXJmYWNlIFJlY3QgZXh0ZW5kcyBEaW1lbnNpb24ge1xuICAgIHg6IG51bWJlclxuICAgIHk6IG51bWJlclxufVxuIl19