@ngutil/layout 0.0.63 → 0.0.64

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,215 +1,70 @@
1
- import { Component, ElementRef, inject, Input, Output, ViewChild } from "@angular/core";
2
- import { BehaviorSubject, combineLatest, map, of, ReplaySubject, shareReplay, switchMap } from "rxjs";
3
- import { coerceBoolAttr, Destructible, FastDOM, NumberWithUnit } from "@ngutil/common";
1
+ import { Component, computed, ElementRef, inject, input, model, viewChild } from "@angular/core";
2
+ import { takeUntilDestroyed, toObservable, toSignal } from "@angular/core/rxjs-interop";
3
+ import { switchMap } from "rxjs";
4
+ import { coerceBoolAttr } from "@ngutil/common";
4
5
  import { DimensionWatcher } from "@ngutil/style";
5
6
  import { L9Range } from "../l9/range";
6
7
  import * as i0 from "@angular/core";
7
8
  const DEFAULT_POSITION = L9Range.coerce("left");
8
- const HIDDEN_SIZE = new NumberWithUnit(0, "px");
9
- const AUTO_SIZE = NumberWithUnit.coerce("auto");
10
- export class DockingPanelComponent extends Destructible {
11
- #dimWatcher;
12
- set positionInput(val) {
13
- const coerced = L9Range.coerce(val);
14
- if (coerced.orient === "rect") {
15
- throw new Error(`Invalid position value: ${val}`);
16
- }
17
- if (!this.position.value.isEq(coerced)) {
18
- this.position.next(coerced);
19
- }
20
- }
21
- set stateInput(val) {
22
- if (this.state.value !== val) {
23
- this.state.next(val);
24
- }
25
- }
26
- set modeInput(val) {
27
- if (this.mode.value !== val) {
28
- this.mode.next(val);
29
- }
30
- }
31
- set fullSizeInput(val) {
32
- const coerced = NumberWithUnit.coerce(val, "px");
33
- if (this.#fullSize.value !== coerced) {
34
- this.#fullSize.next(coerced);
35
- }
36
- }
37
- #fullSize;
38
- set miniSizeInput(val) {
39
- const coerced = NumberWithUnit.coerce(val, "px");
40
- if (this.#miniSize.value !== coerced) {
41
- this.#miniSize.next(coerced);
42
- }
43
- }
44
- #miniSize;
45
- set minimizable(val) {
46
- this.#minimizable = coerceBoolAttr(val);
47
- this.#minimizableAuto = false;
48
- }
49
- get minimizable() {
50
- return this.#minimizable;
51
- }
52
- #minimizable;
53
- #minimizableAuto;
54
- set backdrop(val) {
55
- this.#backdrop = coerceBoolAttr(val);
56
- }
57
- get backdrop() {
58
- return this.#backdrop;
59
- }
60
- #backdrop;
61
- #contentSize;
62
- #autoSize;
63
- // TODO: better animation handling in min -> hidden -> min -> full
9
+ export class DockingPanelComponent {
64
10
  constructor() {
65
- super();
66
- this.el = inject((ElementRef));
67
11
  this.#dimWatcher = inject(DimensionWatcher);
68
- this.position = new BehaviorSubject(DEFAULT_POSITION);
69
- this.state = new BehaviorSubject("full");
70
- this.mode = new BehaviorSubject("rigid");
71
- this.#fullSize = new BehaviorSubject(AUTO_SIZE);
72
- this.#miniSize = new BehaviorSubject(HIDDEN_SIZE);
73
- this.#minimizable = false;
74
- this.#minimizableAuto = true;
75
- this.#backdrop = false;
76
- this.#contentSize = new ReplaySubject(1);
77
- this.#autoSize = combineLatest({
78
- dim: this.#contentSize,
79
- pos: this.position
80
- }).pipe(map(({ dim, pos }) => {
81
- if (pos.orient === "horizontal") {
82
- return dim.height;
83
- }
84
- else {
85
- return dim.width;
86
- }
87
- }), shareReplay(1));
88
- this.fullSize = this.#fullSize.pipe(switchMap(size => {
89
- if (size.unit === "auto") {
90
- return this.#autoSize;
91
- }
92
- else {
93
- return of(size);
94
- }
95
- }), shareReplay(1));
96
- this.miniSize = this.#miniSize.pipe(switchMap(size => {
97
- if (size.unit === "auto") {
98
- return this.#autoSize;
99
- }
100
- else {
101
- return of(size);
102
- }
103
- }), shareReplay(1));
104
- this.changes = combineLatest({
105
- position: this.position,
106
- state: this.state,
107
- mode: this.mode,
108
- fullSize: this.fullSize,
109
- miniSize: this.miniSize,
110
- contentSize: this.#contentSize
12
+ this.el = inject(ElementRef);
13
+ this.position = input(DEFAULT_POSITION, { transform: L9Range.coerce });
14
+ // TODO: linkedSignal
15
+ this.opened = model(false);
16
+ this._opened = computed(() => coerceBoolAttr(this.opened()));
17
+ this.mode = input("rigid");
18
+ this.maxSize = input(undefined);
19
+ this.backdrop = input(false);
20
+ this.gridArea = computed(() => this.position().intoGridArea());
21
+ this.orient = computed(() => this.position().orient);
22
+ this.side = computed(() => {
23
+ const pos = this.position();
24
+ return pos.orient === "horizontal" ? pos.cells[0].v : pos.cells[0].h;
111
25
  });
112
- this.d.sub(this.changes).subscribe(changes => {
113
- if (this.#minimizableAuto) {
114
- this.#minimizable = this.#miniSize.value.value !== 0;
26
+ this.content = viewChild.required("content", { read: ElementRef });
27
+ this.dimension$ = toObservable(this.content).pipe(switchMap(content => this.#dimWatcher.watch(content, "border-box")), takeUntilDestroyed());
28
+ this.dimension = toSignal(this.dimension$);
29
+ this.contentSize = computed(() => {
30
+ const dim = this.dimension();
31
+ if (!dim) {
32
+ return 0;
115
33
  }
116
- FastDOM.setAttributes(this.el.nativeElement, {
117
- state: changes.state,
118
- orient: changes.position.orient,
119
- mode: changes.mode,
120
- side: changes.position.orient === "horizontal" ? changes.position.cells[0].v : changes.position.cells[0].h
121
- });
122
- const isHorizontal = changes.position.orient === "horizontal";
123
- let w = null;
124
- let h = null;
125
- // TODO: when change state from mini -> hidden, currently wrong behavior
126
- // the good behavior is to not gain fullSize ang go to hidden
127
- if (changes.state === "mini") {
128
- if (isHorizontal) {
129
- h = changes.miniSize.unit === "auto" ? changes.contentSize.height : changes.miniSize;
130
- }
131
- else {
132
- w = changes.miniSize.unit === "auto" ? changes.contentSize.width : changes.miniSize;
133
- }
134
- }
135
- else {
136
- if (isHorizontal) {
137
- h = changes.fullSize.unit === "auto" ? changes.contentSize.height : changes.fullSize;
138
- }
139
- else {
140
- w = changes.fullSize.unit === "auto" ? changes.contentSize.width : changes.fullSize;
141
- }
142
- }
143
- FastDOM.setStyle(this.el.nativeElement, {
144
- "--docking-panel-w": w != null ? `${w}` : null,
145
- "--docking-panel-h": h != null ? `${h}` : null,
146
- "--docking-panel-content-w": changes.contentSize.width,
147
- "--docking-panel-content-h": changes.contentSize.height
148
- }, () => FastDOM.setAttributes(this.el.nativeElement, { animate: "" }));
34
+ return this.orient() === "horizontal" ? dim.height : dim.width;
149
35
  });
150
36
  }
151
- ngAfterViewInit() {
152
- this.d
153
- .sub(this.#dimWatcher.watch(this.content, "scroll-box"))
154
- .pipe(map(dim => {
155
- return {
156
- width: new NumberWithUnit(dim.width, "px"),
157
- height: new NumberWithUnit(dim.height, "px")
158
- };
159
- }))
160
- .subscribe(this.#contentSize);
161
- }
37
+ #dimWatcher;
38
+ // readonly backdropSize = computed(() => {
39
+ // const mode = this.backdrop()
40
+ // if (mode === true || mode === "full") {
41
+ // return L9Range.coerce("top:left-bottom:right").intoGridArea()
42
+ // } else if (mode === "panel-size") {
43
+ // const pos = this.position()
44
+ // if (pos.orient === "horizontal") {
45
+ // }
46
+ // return ""
47
+ // }
48
+ // return null
49
+ // })
162
50
  open() {
163
- this.state.next("full");
51
+ this.opened.set(true);
164
52
  }
165
53
  close() {
166
- this.state.next("hidden");
167
- }
168
- minimize() {
169
- if (this.minimizable) {
170
- this.state.next("mini");
171
- }
54
+ this.opened.set(false);
172
55
  }
173
56
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: DockingPanelComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
174
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.6", type: DockingPanelComponent, isStandalone: true, selector: "nu-docking-panel", inputs: { positionInput: ["position", "positionInput"], stateInput: ["state", "stateInput"], modeInput: ["mode", "modeInput"], fullSizeInput: ["fullSize", "fullSizeInput"], miniSizeInput: ["miniSize", "miniSizeInput"], minimizable: "minimizable", backdrop: "backdrop" }, outputs: { state: "stateChanges" }, viewQueries: [{ propertyName: "content", first: true, predicate: ["content"], descendants: true, read: ElementRef, static: true }], exportAs: ["nuDockingPanel"], usesInheritance: true, ngImport: i0, template: `
175
- <div class="nu-docking-wrapper" #content>
176
- <ng-content></ng-content>
177
- </div>
178
- `, isInline: true, styles: [":host{---docking-panel-t: var(--docking-panel-t, auto);---docking-panel-r: var(--docking-panel-r, auto);---docking-panel-b: var(--docking-panel-b, auto);---docking-panel-l: var(--docking-panel-l, auto);---docking-panel-w: var(--docking-panel-w, auto);---docking-panel-h: var(--docking-panel-h, auto);---docking-panel-content-w: var(--docking-panel-content-w, var(---docking-panel-w));---docking-panel-content-g: var(--docking-panel-content-g, var(---docking-panel-h));display:flex;flex-flow:column nowrap;align-items:stretch;position:absolute;box-sizing:border-box;overflow:hidden;top:var(---docking-panel-t);right:var(---docking-panel-r);bottom:var(---docking-panel-b);left:var(---docking-panel-l);width:var(---docking-panel-w);height:var(---docking-panel-h)}:host[animate]{transition:transform var(---docking-layout-anim-duration) var(---docking-layout-anim-ease),width var(---docking-layout-anim-duration) var(---docking-layout-anim-ease),height var(---docking-layout-anim-duration) var(---docking-layout-anim-ease)}:host[side=top],:host[side=left]{---docking-panel-t-hide: -100%;---docking-panel-t-visible: 0%}:host[side=bottom],:host[side=right]{---docking-panel-t-hide: 100%;---docking-panel-t-visible: 0%}:host[state=hidden]{animation:var(---docking-layout-anim-duration) var(---docking-layout-anim-ease) hide;animation-fill-mode:forwards}:host[state=hidden][orient=horizontal]{transform:translateY(var(---docking-panel-t-hide))}:host[state=hidden][orient=vertical]{transform:translate(var(---docking-panel-t-hide))}:host:not([state=hidden]){visibility:visible}:host:not([state=hidden])[orient=horizontal]{transform:translateY(var(---docking-panel-t-visible))}:host:not([state=hidden])[orient=vertical]{transform:translate(var(---docking-panel-t-visible))}:host .nu-docking-wrapper{display:flex;flex-direction:column;align-items:stretch}:host[orient=horizontal] .nu-docking-wrapper{width:100%;min-width:100%;max-width:100%}:host[orient=vertical] .nu-docking-wrapper{height:100%;min-height:100%;max-height:100%}@keyframes hide{99%{visibility:visible}to{visibility:hidden}}\n"] }); }
57
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.2.0", version: "18.2.6", type: DockingPanelComponent, isStandalone: true, selector: "nu-docking-panel", inputs: { position: { classPropertyName: "position", publicName: "position", isSignal: true, isRequired: false, transformFunction: null }, opened: { classPropertyName: "opened", publicName: "opened", isSignal: true, isRequired: false, transformFunction: null }, mode: { classPropertyName: "mode", publicName: "mode", isSignal: true, isRequired: false, transformFunction: null }, maxSize: { classPropertyName: "maxSize", publicName: "maxSize", isSignal: true, isRequired: false, transformFunction: null }, backdrop: { classPropertyName: "backdrop", publicName: "backdrop", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { opened: "openedChange" }, host: { properties: { "style.grid-area": "gridArea()", "style.--nudp-content-size.px": "contentSize()", "attr.state": "_opened() ? 'opened' : 'closed'", "attr.orient": "orient()", "attr.side": "side()", "attr.mode": "mode()" } }, viewQueries: [{ propertyName: "content", first: true, predicate: ["content"], descendants: true, read: ElementRef, isSignal: true }], exportAs: ["nuDockingPanel"], ngImport: i0, template: `<div class="wrapper" #content><ng-content /></div>`, isInline: true, styles: [":host{--nudp-tx: 0;--nudp-ty: 0;--nudp-t: auto;--nudp-r: auto;--nudp-b: auto;--nudp-l: auto;display:block;position:relative;z-index:100}:host .wrapper{all:inherit;display:grid;grid-template-columns:auto;grid-template-rows:auto;align-items:stretch;justify-items:stretch;position:absolute;top:var(--nudp-t);right:var(--nudp-r);bottom:var(--nudp-b);left:var(--nudp-l);width:auto;height:auto;transition:transform .2s cubic-bezier(0,0,.2,1);transform:translate(var(--nudp-tx),var(--nudp-ty))}:host[mode=rigid]{overflow:clip}:host[mode=rigid][orient=horizontal]{transition:height .2s cubic-bezier(0,0,.2,1);height:var(--nudp-content-size)}:host[mode=rigid][orient=horizontal][state=closed]{height:0px}:host[mode=rigid][orient=vertical]{transition:width .2s cubic-bezier(0,0,.2,1);width:var(--nudp-content-size)}:host[mode=rigid][orient=vertical][state=closed]{width:0}:host[mode=over]{z-index:300}:host[orient=horizontal]{--nudp-l: 0px;--nudp-r: 0px}:host[orient=horizontal][side=top]{--nudp-t: 0px}:host[orient=horizontal][side=top][state=closed]{--nudp-ty: -100%}:host[orient=horizontal][side=bottom]{--nudp-b: 0px}:host[orient=horizontal][side=bottom][state=closed]{--nudp-ty: 100%}:host[orient=horizontal]:not([mode=rigid]){height:0px}:host[orient=vertical]{--nudp-t: 0px;--nudp-b: 0px}:host[orient=vertical][side=left]{--nudp-l: 0px}:host[orient=vertical][side=left][state=closed]{--nudp-tx: -100%}:host[orient=vertical][side=right]{--nudp-r: 0px}:host[orient=vertical][side=right][state=closed]{--nudp-tx: 100%}:host[orient=vertical]:not([mode=rigid]){width:0px}\n"] }); }
179
58
  }
180
59
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.6", ngImport: i0, type: DockingPanelComponent, decorators: [{
181
60
  type: Component,
182
- args: [{ standalone: true, selector: "nu-docking-panel", exportAs: "nuDockingPanel", template: `
183
- <div class="nu-docking-wrapper" #content>
184
- <ng-content></ng-content>
185
- </div>
186
- `, styles: [":host{---docking-panel-t: var(--docking-panel-t, auto);---docking-panel-r: var(--docking-panel-r, auto);---docking-panel-b: var(--docking-panel-b, auto);---docking-panel-l: var(--docking-panel-l, auto);---docking-panel-w: var(--docking-panel-w, auto);---docking-panel-h: var(--docking-panel-h, auto);---docking-panel-content-w: var(--docking-panel-content-w, var(---docking-panel-w));---docking-panel-content-g: var(--docking-panel-content-g, var(---docking-panel-h));display:flex;flex-flow:column nowrap;align-items:stretch;position:absolute;box-sizing:border-box;overflow:hidden;top:var(---docking-panel-t);right:var(---docking-panel-r);bottom:var(---docking-panel-b);left:var(---docking-panel-l);width:var(---docking-panel-w);height:var(---docking-panel-h)}:host[animate]{transition:transform var(---docking-layout-anim-duration) var(---docking-layout-anim-ease),width var(---docking-layout-anim-duration) var(---docking-layout-anim-ease),height var(---docking-layout-anim-duration) var(---docking-layout-anim-ease)}:host[side=top],:host[side=left]{---docking-panel-t-hide: -100%;---docking-panel-t-visible: 0%}:host[side=bottom],:host[side=right]{---docking-panel-t-hide: 100%;---docking-panel-t-visible: 0%}:host[state=hidden]{animation:var(---docking-layout-anim-duration) var(---docking-layout-anim-ease) hide;animation-fill-mode:forwards}:host[state=hidden][orient=horizontal]{transform:translateY(var(---docking-panel-t-hide))}:host[state=hidden][orient=vertical]{transform:translate(var(---docking-panel-t-hide))}:host:not([state=hidden]){visibility:visible}:host:not([state=hidden])[orient=horizontal]{transform:translateY(var(---docking-panel-t-visible))}:host:not([state=hidden])[orient=vertical]{transform:translate(var(---docking-panel-t-visible))}:host .nu-docking-wrapper{display:flex;flex-direction:column;align-items:stretch}:host[orient=horizontal] .nu-docking-wrapper{width:100%;min-width:100%;max-width:100%}:host[orient=vertical] .nu-docking-wrapper{height:100%;min-height:100%;max-height:100%}@keyframes hide{99%{visibility:visible}to{visibility:hidden}}\n"] }]
187
- }], ctorParameters: () => [], propDecorators: { content: [{
188
- type: ViewChild,
189
- args: ["content", { read: ElementRef, static: true }]
190
- }], positionInput: [{
191
- type: Input,
192
- args: ["position"]
193
- }], stateInput: [{
194
- type: Input,
195
- args: ["state"]
196
- }], state: [{
197
- type: Output,
198
- args: ["stateChanges"]
199
- }], modeInput: [{
200
- type: Input,
201
- args: ["mode"]
202
- }], fullSizeInput: [{
203
- type: Input,
204
- args: ["fullSize"]
205
- }], miniSizeInput: [{
206
- type: Input,
207
- args: ["miniSize"]
208
- }], minimizable: [{
209
- type: Input,
210
- args: ["minimizable"]
211
- }], backdrop: [{
212
- type: Input,
213
- args: ["backdrop"]
214
- }] } });
215
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docking-panel.component.js","sourceRoot":"","sources":["../../../../../packages/layout/src/docking/docking-panel.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, ElementRef, inject, Input, Output, ViewChild } from \"@angular/core\"\n\nimport { BehaviorSubject, combineLatest, map, Observable, of, ReplaySubject, shareReplay, switchMap } from \"rxjs\"\n\nimport {\n    BooleanInput,\n    coerceBoolAttr,\n    Destructible,\n    FastDOM,\n    NumberWithUnit,\n    NumberWithUnitInput\n} from \"@ngutil/common\"\nimport { DimensionWatcher } from \"@ngutil/style\"\n\nimport { L9Range, L9RangeName } from \"../l9/range\"\n\nexport type DockingPanelState = \"full\" | \"mini\" | \"hidden\"\nexport type DockingPanelMode = \"over\" | \"push\" | \"rigid\"\n\nconst DEFAULT_POSITION = L9Range.coerce(\"left\")\nconst HIDDEN_SIZE = new NumberWithUnit(0, \"px\")\nconst AUTO_SIZE = NumberWithUnit.coerce(\"auto\")\n\n@Component({\n    standalone: true,\n    selector: \"nu-docking-panel\",\n    exportAs: \"nuDockingPanel\",\n    styleUrl: \"./docking-panel.component.scss\",\n    template: `\n        <div class=\"nu-docking-wrapper\" #content>\n            <ng-content></ng-content>\n        </div>\n    `\n})\nexport class DockingPanelComponent extends Destructible implements AfterViewInit {\n    readonly el = inject(ElementRef<HTMLElement>)\n    readonly #dimWatcher = inject(DimensionWatcher)\n    @ViewChild(\"content\", { read: ElementRef, static: true })\n    readonly content!: ElementRef<HTMLElement>\n\n    @Input(\"position\")\n    set positionInput(val: L9Range | L9RangeName) {\n        const coerced = L9Range.coerce(val)\n        if (coerced.orient === \"rect\") {\n            throw new Error(`Invalid position value: ${val}`)\n        }\n\n        if (!this.position.value.isEq(coerced)) {\n            this.position.next(coerced)\n        }\n    }\n    readonly position = new BehaviorSubject<L9Range>(DEFAULT_POSITION)\n\n    @Input(\"state\")\n    set stateInput(val: DockingPanelState) {\n        if (this.state.value !== val) {\n            this.state.next(val)\n        }\n    }\n    @Output(\"stateChanges\")\n    readonly state = new BehaviorSubject<DockingPanelState>(\"full\")\n\n    @Input(\"mode\")\n    set modeInput(val: DockingPanelMode) {\n        if (this.mode.value !== val) {\n            this.mode.next(val)\n        }\n    }\n    readonly mode = new BehaviorSubject<DockingPanelMode>(\"rigid\")\n\n    @Input(\"fullSize\")\n    set fullSizeInput(val: NumberWithUnitInput) {\n        const coerced = NumberWithUnit.coerce(val, \"px\")\n        if (this.#fullSize.value !== coerced) {\n            this.#fullSize.next(coerced)\n        }\n    }\n    readonly #fullSize = new BehaviorSubject<NumberWithUnit>(AUTO_SIZE)\n\n    @Input(\"miniSize\")\n    set miniSizeInput(val: NumberWithUnitInput) {\n        const coerced = NumberWithUnit.coerce(val, \"px\")\n        if (this.#miniSize.value !== coerced) {\n            this.#miniSize.next(coerced)\n        }\n    }\n    readonly #miniSize = new BehaviorSubject<NumberWithUnit>(HIDDEN_SIZE)\n\n    @Input(\"minimizable\")\n    set minimizable(val: BooleanInput) {\n        this.#minimizable = coerceBoolAttr(val)\n        this.#minimizableAuto = false\n    }\n    get minimizable(): boolean {\n        return this.#minimizable\n    }\n    #minimizable: boolean = false\n    #minimizableAuto: boolean = true\n\n    @Input(\"backdrop\")\n    set backdrop(val: BooleanInput) {\n        this.#backdrop = coerceBoolAttr(val)\n    }\n    get backdrop(): boolean {\n        return this.#backdrop\n    }\n    #backdrop: boolean = false\n\n    readonly #contentSize = new ReplaySubject<{ width: NumberWithUnit; height: NumberWithUnit }>(1)\n\n    readonly #autoSize = combineLatest({\n        dim: this.#contentSize,\n        pos: this.position\n    }).pipe(\n        map(({ dim, pos }) => {\n            if (pos.orient === \"horizontal\") {\n                return dim.height\n            } else {\n                return dim.width\n            }\n        }),\n        shareReplay(1)\n    )\n\n    readonly fullSize = this.#fullSize.pipe(\n        switchMap(size => {\n            if (size.unit === \"auto\") {\n                return this.#autoSize\n            } else {\n                return of(size)\n            }\n        }),\n        shareReplay(1)\n    )\n\n    readonly miniSize = this.#miniSize.pipe(\n        switchMap(size => {\n            if (size.unit === \"auto\") {\n                return this.#autoSize\n            } else {\n                return of(size)\n            }\n        }),\n        shareReplay(1)\n    )\n\n    readonly changes = combineLatest({\n        position: this.position,\n        state: this.state,\n        mode: this.mode,\n        fullSize: this.fullSize,\n        miniSize: this.miniSize,\n        contentSize: this.#contentSize\n    })\n\n    // TODO: better animation handling in min -> hidden -> min -> full\n    constructor() {\n        super()\n\n        this.d.sub(this.changes).subscribe(changes => {\n            if (this.#minimizableAuto) {\n                this.#minimizable = this.#miniSize.value.value !== 0\n            }\n\n            FastDOM.setAttributes(this.el.nativeElement, {\n                state: changes.state,\n                orient: changes.position.orient,\n                mode: changes.mode,\n                side:\n                    changes.position.orient === \"horizontal\" ? changes.position.cells[0].v : changes.position.cells[0].h\n            })\n\n            const isHorizontal = changes.position.orient === \"horizontal\"\n            let w = null\n            let h = null\n\n            // TODO: when change state from mini -> hidden, currently wrong behavior\n            // the good behavior is to not gain fullSize ang go to hidden\n            if (changes.state === \"mini\") {\n                if (isHorizontal) {\n                    h = changes.miniSize.unit === \"auto\" ? changes.contentSize.height : changes.miniSize\n                } else {\n                    w = changes.miniSize.unit === \"auto\" ? changes.contentSize.width : changes.miniSize\n                }\n            } else {\n                if (isHorizontal) {\n                    h = changes.fullSize.unit === \"auto\" ? changes.contentSize.height : changes.fullSize\n                } else {\n                    w = changes.fullSize.unit === \"auto\" ? changes.contentSize.width : changes.fullSize\n                }\n            }\n\n            FastDOM.setStyle(\n                this.el.nativeElement,\n                {\n                    \"--docking-panel-w\": w != null ? `${w}` : null,\n                    \"--docking-panel-h\": h != null ? `${h}` : null,\n                    \"--docking-panel-content-w\": changes.contentSize.width,\n                    \"--docking-panel-content-h\": changes.contentSize.height\n                },\n                () => FastDOM.setAttributes(this.el.nativeElement, { animate: \"\" })\n            )\n        })\n    }\n\n    ngAfterViewInit(): void {\n        this.d\n            .sub(this.#dimWatcher.watch(this.content, \"scroll-box\"))\n            .pipe(\n                map(dim => {\n                    return {\n                        width: new NumberWithUnit(dim.width, \"px\"),\n                        height: new NumberWithUnit(dim.height, \"px\")\n                    }\n                })\n            )\n            .subscribe(this.#contentSize)\n    }\n\n    open() {\n        this.state.next(\"full\")\n    }\n\n    close() {\n        this.state.next(\"hidden\")\n    }\n\n    minimize() {\n        if (this.minimizable) {\n            this.state.next(\"mini\")\n        }\n    }\n}\n\ntype ChangesObservable = typeof DockingPanelComponent.prototype.changes\nexport type DockingPanelChanges = ChangesObservable extends Observable<infer T> ? T : never\n"]}
61
+ args: [{ selector: "nu-docking-panel", exportAs: "nuDockingPanel", standalone: true, host: {
62
+ "[style.grid-area]": "gridArea()",
63
+ "[style.--nudp-content-size.px]": "contentSize()",
64
+ "[attr.state]": "_opened() ? 'opened' : 'closed'",
65
+ "[attr.orient]": "orient()",
66
+ "[attr.side]": "side()",
67
+ "[attr.mode]": "mode()"
68
+ }, template: `<div class="wrapper" #content><ng-content /></div>`, styles: [":host{--nudp-tx: 0;--nudp-ty: 0;--nudp-t: auto;--nudp-r: auto;--nudp-b: auto;--nudp-l: auto;display:block;position:relative;z-index:100}:host .wrapper{all:inherit;display:grid;grid-template-columns:auto;grid-template-rows:auto;align-items:stretch;justify-items:stretch;position:absolute;top:var(--nudp-t);right:var(--nudp-r);bottom:var(--nudp-b);left:var(--nudp-l);width:auto;height:auto;transition:transform .2s cubic-bezier(0,0,.2,1);transform:translate(var(--nudp-tx),var(--nudp-ty))}:host[mode=rigid]{overflow:clip}:host[mode=rigid][orient=horizontal]{transition:height .2s cubic-bezier(0,0,.2,1);height:var(--nudp-content-size)}:host[mode=rigid][orient=horizontal][state=closed]{height:0px}:host[mode=rigid][orient=vertical]{transition:width .2s cubic-bezier(0,0,.2,1);width:var(--nudp-content-size)}:host[mode=rigid][orient=vertical][state=closed]{width:0}:host[mode=over]{z-index:300}:host[orient=horizontal]{--nudp-l: 0px;--nudp-r: 0px}:host[orient=horizontal][side=top]{--nudp-t: 0px}:host[orient=horizontal][side=top][state=closed]{--nudp-ty: -100%}:host[orient=horizontal][side=bottom]{--nudp-b: 0px}:host[orient=horizontal][side=bottom][state=closed]{--nudp-ty: 100%}:host[orient=horizontal]:not([mode=rigid]){height:0px}:host[orient=vertical]{--nudp-t: 0px;--nudp-b: 0px}:host[orient=vertical][side=left]{--nudp-l: 0px}:host[orient=vertical][side=left][state=closed]{--nudp-tx: -100%}:host[orient=vertical][side=right]{--nudp-r: 0px}:host[orient=vertical][side=right][state=closed]{--nudp-tx: 100%}:host[orient=vertical]:not([mode=rigid]){width:0px}\n"] }]
69
+ }] });
70
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG9ja2luZy1wYW5lbC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9sYXlvdXQvc3JjL2RvY2tpbmcvZG9ja2luZy1wYW5lbC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsS0FBSyxFQUFFLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQTtBQUNoRyxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsWUFBWSxFQUFFLFFBQVEsRUFBRSxNQUFNLDRCQUE0QixDQUFBO0FBRXZGLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxNQUFNLENBQUE7QUFFaEMsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGdCQUFnQixDQUFBO0FBQy9DLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGVBQWUsQ0FBQTtBQUVoRCxPQUFPLEVBQUUsT0FBTyxFQUFFLE1BQU0sYUFBYSxDQUFBOztBQU1yQyxNQUFNLGdCQUFnQixHQUFHLE9BQU8sQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUE7QUFpQi9DLE1BQU0sT0FBTyxxQkFBcUI7SUFmbEM7UUFnQmEsZ0JBQVcsR0FBRyxNQUFNLENBQUMsZ0JBQWdCLENBQUMsQ0FBQTtRQUN0QyxPQUFFLEdBQUcsTUFBTSxDQUFDLFVBQVUsQ0FBQyxDQUFBO1FBRXZCLGFBQVEsR0FBRyxLQUFLLENBQUMsZ0JBQWdCLEVBQUUsRUFBRSxTQUFTLEVBQUUsT0FBTyxDQUFDLE1BQU0sRUFBRSxDQUFDLENBQUE7UUFDMUUscUJBQXFCO1FBQ1osV0FBTSxHQUFHLEtBQUssQ0FBVSxLQUFLLENBQUMsQ0FBQTtRQUM5QixZQUFPLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLGNBQWMsQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQyxDQUFBO1FBQ3ZELFNBQUksR0FBRyxLQUFLLENBQW1CLE9BQU8sQ0FBQyxDQUFBO1FBQ3ZDLFlBQU8sR0FBRyxLQUFLLENBQTRCLFNBQVMsQ0FBQyxDQUFBO1FBQ3JELGFBQVEsR0FBRyxLQUFLLENBQWUsS0FBSyxDQUFDLENBQUE7UUFFckMsYUFBUSxHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUMsWUFBWSxFQUFFLENBQUMsQ0FBQTtRQUN6RCxXQUFNLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQyxNQUFNLENBQUMsQ0FBQTtRQUMvQyxTQUFJLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRTtZQUMxQixNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUE7WUFDM0IsT0FBTyxHQUFHLENBQUMsTUFBTSxLQUFLLFlBQVksQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFBO1FBQ3hFLENBQUMsQ0FBQyxDQUFBO1FBRU8sWUFBTyxHQUFHLFNBQVMsQ0FBQyxRQUFRLENBQUMsU0FBUyxFQUFFLEVBQUUsSUFBSSxFQUFFLFVBQVUsRUFBRSxDQUFDLENBQUE7UUFFN0QsZUFBVSxHQUFHLFlBQVksQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUMsSUFBSSxDQUNqRCxTQUFTLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLEtBQUssQ0FBQyxPQUFPLEVBQUUsWUFBWSxDQUFDLENBQUMsRUFDbkUsa0JBQWtCLEVBQUUsQ0FDdkIsQ0FBQTtRQUNRLGNBQVMsR0FBRyxRQUFRLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFBO1FBRXJDLGdCQUFXLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRTtZQUNqQyxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUE7WUFDNUIsSUFBSSxDQUFDLEdBQUcsRUFBRSxDQUFDO2dCQUNQLE9BQU8sQ0FBQyxDQUFBO1lBQ1osQ0FBQztZQUNELE9BQU8sSUFBSSxDQUFDLE1BQU0sRUFBRSxLQUFLLFlBQVksQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQTtRQUNsRSxDQUFDLENBQUMsQ0FBQTtLQXdCTDtJQXhEWSxXQUFXLENBQTJCO0lBa0MvQywyQ0FBMkM7SUFDM0MsbUNBQW1DO0lBQ25DLDhDQUE4QztJQUM5Qyx3RUFBd0U7SUFDeEUsMENBQTBDO0lBQzFDLHNDQUFzQztJQUN0Qyw2Q0FBNkM7SUFFN0MsWUFBWTtJQUVaLG9CQUFvQjtJQUNwQixRQUFRO0lBQ1Isa0JBQWtCO0lBQ2xCLEtBQUs7SUFFTCxJQUFJO1FBQ0EsSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUE7SUFDekIsQ0FBQztJQUVELEtBQUs7UUFDRCxJQUFJLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQTtJQUMxQixDQUFDOzhHQXhEUSxxQkFBcUI7a0dBQXJCLHFCQUFxQixraUNBbUIyQixVQUFVLDJFQXJCekQsb0RBQW9EOzsyRkFFckQscUJBQXFCO2tCQWZqQyxTQUFTOytCQUNJLGtCQUFrQixZQUNsQixnQkFBZ0IsY0FDZCxJQUFJLFFBQ1Y7d0JBQ0YsbUJBQW1CLEVBQUUsWUFBWTt3QkFDakMsZ0NBQWdDLEVBQUUsZUFBZTt3QkFDakQsY0FBYyxFQUFFLGlDQUFpQzt3QkFDakQsZUFBZSxFQUFFLFVBQVU7d0JBQzNCLGFBQWEsRUFBRSxRQUFRO3dCQUN2QixhQUFhLEVBQUUsUUFBUTtxQkFDMUIsWUFFUyxvREFBb0QiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIGNvbXB1dGVkLCBFbGVtZW50UmVmLCBpbmplY3QsIGlucHV0LCBtb2RlbCwgdmlld0NoaWxkIH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIlxuaW1wb3J0IHsgdGFrZVVudGlsRGVzdHJveWVkLCB0b09ic2VydmFibGUsIHRvU2lnbmFsIH0gZnJvbSBcIkBhbmd1bGFyL2NvcmUvcnhqcy1pbnRlcm9wXCJcblxuaW1wb3J0IHsgc3dpdGNoTWFwIH0gZnJvbSBcInJ4anNcIlxuXG5pbXBvcnQgeyBjb2VyY2VCb29sQXR0ciB9IGZyb20gXCJAbmd1dGlsL2NvbW1vblwiXG5pbXBvcnQgeyBEaW1lbnNpb25XYXRjaGVyIH0gZnJvbSBcIkBuZ3V0aWwvc3R5bGVcIlxuXG5pbXBvcnQgeyBMOVJhbmdlIH0gZnJvbSBcIi4uL2w5L3JhbmdlXCJcblxuZXhwb3J0IHR5cGUgRG9ja2luZ1BhbmVsU3RhdGUgPSBcImZ1bGxcIiB8IFwibWluaVwiIHwgXCJoaWRkZW5cIlxuZXhwb3J0IHR5cGUgRG9ja2luZ1BhbmVsTW9kZSA9IFwib3ZlclwiIHwgXCJwdXNoXCIgfCBcInJpZ2lkXCJcbmV4cG9ydCB0eXBlIEJhY2tkcm9wTW9kZSA9IGJvb2xlYW4gfCBcImZ1bGxcIiB8IFwicGFuZWwtc2l6ZVwiXG5cbmNvbnN0IERFRkFVTFRfUE9TSVRJT04gPSBMOVJhbmdlLmNvZXJjZShcImxlZnRcIilcblxuQENvbXBvbmVudCh7XG4gICAgc2VsZWN0b3I6IFwibnUtZG9ja2luZy1wYW5lbFwiLFxuICAgIGV4cG9ydEFzOiBcIm51RG9ja2luZ1BhbmVsXCIsXG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBob3N0OiB7XG4gICAgICAgIFwiW3N0eWxlLmdyaWQtYXJlYV1cIjogXCJncmlkQXJlYSgpXCIsXG4gICAgICAgIFwiW3N0eWxlLi0tbnVkcC1jb250ZW50LXNpemUucHhdXCI6IFwiY29udGVudFNpemUoKVwiLFxuICAgICAgICBcIlthdHRyLnN0YXRlXVwiOiBcIl9vcGVuZWQoKSA/ICdvcGVuZWQnIDogJ2Nsb3NlZCdcIixcbiAgICAgICAgXCJbYXR0ci5vcmllbnRdXCI6IFwib3JpZW50KClcIixcbiAgICAgICAgXCJbYXR0ci5zaWRlXVwiOiBcInNpZGUoKVwiLFxuICAgICAgICBcIlthdHRyLm1vZGVdXCI6IFwibW9kZSgpXCJcbiAgICB9LFxuICAgIHN0eWxlVXJsOiBcIi4vZG9ja2luZy1wYW5lbC5jb21wb25lbnQuc2Nzc1wiLFxuICAgIHRlbXBsYXRlOiBgPGRpdiBjbGFzcz1cIndyYXBwZXJcIiAjY29udGVudD48bmctY29udGVudCAvPjwvZGl2PmBcbn0pXG5leHBvcnQgY2xhc3MgRG9ja2luZ1BhbmVsQ29tcG9uZW50IHtcbiAgICByZWFkb25seSAjZGltV2F0Y2hlciA9IGluamVjdChEaW1lbnNpb25XYXRjaGVyKVxuICAgIHJlYWRvbmx5IGVsID0gaW5qZWN0KEVsZW1lbnRSZWYpXG5cbiAgICByZWFkb25seSBwb3NpdGlvbiA9IGlucHV0KERFRkFVTFRfUE9TSVRJT04sIHsgdHJhbnNmb3JtOiBMOVJhbmdlLmNvZXJjZSB9KVxuICAgIC8vIFRPRE86IGxpbmtlZFNpZ25hbFxuICAgIHJlYWRvbmx5IG9wZW5lZCA9IG1vZGVsPGJvb2xlYW4+KGZhbHNlKVxuICAgIHJlYWRvbmx5IF9vcGVuZWQgPSBjb21wdXRlZCgoKSA9PiBjb2VyY2VCb29sQXR0cih0aGlzLm9wZW5lZCgpKSlcbiAgICByZWFkb25seSBtb2RlID0gaW5wdXQ8RG9ja2luZ1BhbmVsTW9kZT4oXCJyaWdpZFwiKVxuICAgIHJlYWRvbmx5IG1heFNpemUgPSBpbnB1dDxudW1iZXIgfCB1bmRlZmluZWQgfCBudWxsPih1bmRlZmluZWQpXG4gICAgcmVhZG9ubHkgYmFja2Ryb3AgPSBpbnB1dDxCYWNrZHJvcE1vZGU+KGZhbHNlKVxuXG4gICAgcmVhZG9ubHkgZ3JpZEFyZWEgPSBjb21wdXRlZCgoKSA9PiB0aGlzLnBvc2l0aW9uKCkuaW50b0dyaWRBcmVhKCkpXG4gICAgcmVhZG9ubHkgb3JpZW50ID0gY29tcHV0ZWQoKCkgPT4gdGhpcy5wb3NpdGlvbigpLm9yaWVudClcbiAgICByZWFkb25seSBzaWRlID0gY29tcHV0ZWQoKCkgPT4ge1xuICAgICAgICBjb25zdCBwb3MgPSB0aGlzLnBvc2l0aW9uKClcbiAgICAgICAgcmV0dXJuIHBvcy5vcmllbnQgPT09IFwiaG9yaXpvbnRhbFwiID8gcG9zLmNlbGxzWzBdLnYgOiBwb3MuY2VsbHNbMF0uaFxuICAgIH0pXG5cbiAgICByZWFkb25seSBjb250ZW50ID0gdmlld0NoaWxkLnJlcXVpcmVkKFwiY29udGVudFwiLCB7IHJlYWQ6IEVsZW1lbnRSZWYgfSlcblxuICAgIHJlYWRvbmx5IGRpbWVuc2lvbiQgPSB0b09ic2VydmFibGUodGhpcy5jb250ZW50KS5waXBlKFxuICAgICAgICBzd2l0Y2hNYXAoY29udGVudCA9PiB0aGlzLiNkaW1XYXRjaGVyLndhdGNoKGNvbnRlbnQsIFwiYm9yZGVyLWJveFwiKSksXG4gICAgICAgIHRha2VVbnRpbERlc3Ryb3llZCgpXG4gICAgKVxuICAgIHJlYWRvbmx5IGRpbWVuc2lvbiA9IHRvU2lnbmFsKHRoaXMuZGltZW5zaW9uJClcblxuICAgIHJlYWRvbmx5IGNvbnRlbnRTaXplID0gY29tcHV0ZWQoKCkgPT4ge1xuICAgICAgICBjb25zdCBkaW0gPSB0aGlzLmRpbWVuc2lvbigpXG4gICAgICAgIGlmICghZGltKSB7XG4gICAgICAgICAgICByZXR1cm4gMFxuICAgICAgICB9XG4gICAgICAgIHJldHVybiB0aGlzLm9yaWVudCgpID09PSBcImhvcml6b250YWxcIiA/IGRpbS5oZWlnaHQgOiBkaW0ud2lkdGhcbiAgICB9KVxuXG4gICAgLy8gcmVhZG9ubHkgYmFja2Ryb3BTaXplID0gY29tcHV0ZWQoKCkgPT4ge1xuICAgIC8vICAgICBjb25zdCBtb2RlID0gdGhpcy5iYWNrZHJvcCgpXG4gICAgLy8gICAgIGlmIChtb2RlID09PSB0cnVlIHx8IG1vZGUgPT09IFwiZnVsbFwiKSB7XG4gICAgLy8gICAgICAgICByZXR1cm4gTDlSYW5nZS5jb2VyY2UoXCJ0b3A6bGVmdC1ib3R0b206cmlnaHRcIikuaW50b0dyaWRBcmVhKClcbiAgICAvLyAgICAgfSBlbHNlIGlmIChtb2RlID09PSBcInBhbmVsLXNpemVcIikge1xuICAgIC8vICAgICAgICAgY29uc3QgcG9zID0gdGhpcy5wb3NpdGlvbigpXG4gICAgLy8gICAgICAgICBpZiAocG9zLm9yaWVudCA9PT0gXCJob3Jpem9udGFsXCIpIHtcblxuICAgIC8vICAgICAgICAgfVxuXG4gICAgLy8gICAgICAgICByZXR1cm4gXCJcIlxuICAgIC8vICAgICB9XG4gICAgLy8gICAgIHJldHVybiBudWxsXG4gICAgLy8gfSlcblxuICAgIG9wZW4oKSB7XG4gICAgICAgIHRoaXMub3BlbmVkLnNldCh0cnVlKVxuICAgIH1cblxuICAgIGNsb3NlKCkge1xuICAgICAgICB0aGlzLm9wZW5lZC5zZXQoZmFsc2UpXG4gICAgfVxufVxuIl19
@@ -1,26 +1,26 @@
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
1
  const vertical = ["top", "middle", "bottom"];
11
2
  const horizontal = ["left", "center", "right"];
3
+ export const L9GridTopLeft = { row: 1, col: 1 };
12
4
  export class L9Cell {
13
5
  static coerce(value) {
14
- const [v, h] = value.split(":");
15
- if (vertical.includes(v) && horizontal.includes(h)) {
16
- return new L9Cell(v, h);
6
+ const [v1, v2] = value.split(":");
7
+ const v = vertical.includes(v1) ? v1 : v2;
8
+ const h = horizontal.includes(v1) ? v1 : v2;
9
+ if (v === h) {
10
+ throw new Error(`Invalid cell value: ${value}`);
17
11
  }
18
- throw new Error(`Invalid cell value: ${value}`);
12
+ return new L9Cell(v, h);
19
13
  }
20
14
  constructor(v, h) {
21
15
  this.v = v;
22
16
  this.h = h;
23
17
  }
18
+ intoGridArea(gridTopLeft = L9GridTopLeft) {
19
+ return `${gridTopLeft.row + vertical.indexOf(this.v)}/${gridTopLeft.col + horizontal.indexOf(this.h)}`;
20
+ }
21
+ isEq(other) {
22
+ return this.v === other.v && this.h === other.h;
23
+ }
24
24
  }
25
25
  export class L9Range {
26
26
  static coerce(value) {
@@ -50,18 +50,40 @@ export class L9Range {
50
50
  }
51
51
  return false;
52
52
  }
53
+ intoGridArea(gridTopLeft = L9GridTopLeft) {
54
+ const start = this.cells[0];
55
+ const end = this.cells[this.cells.length - 1];
56
+ const endTopLeft = { row: gridTopLeft.row + 1, col: gridTopLeft.col + 1 };
57
+ return `${start.intoGridArea(gridTopLeft)}/${end.intoGridArea(endTopLeft)}`;
58
+ }
59
+ intoRect() {
60
+ const start = this.cells[0];
61
+ const end = this.cells[this.cells.length - 1];
62
+ const x = horizontal.indexOf(start.h);
63
+ const y = vertical.indexOf(start.v);
64
+ return { x, y, width: horizontal.indexOf(end.h) - x + 1, height: vertical.indexOf(end.v) - y + 1 };
65
+ }
53
66
  #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) {
67
+ const rect = this.intoRect();
68
+ if (rect.width === rect.height) {
69
+ // corner
70
+ if (rect.x === rect.y) {
71
+ return "vertical";
72
+ }
73
+ if (rect.x === 0 || rect.x === 2) {
74
+ return "vertical";
75
+ }
76
+ else if (rect.y === 0 || rect.y === 2) {
77
+ return "horizontal";
78
+ }
79
+ }
80
+ else if (rect.width > rect.height) {
57
81
  return "horizontal";
58
82
  }
59
- else if (sh === eh) {
83
+ else if (rect.height > rect.width) {
60
84
  return "vertical";
61
85
  }
62
- else {
63
- return "rect";
64
- }
86
+ return "vertical";
65
87
  }
66
88
  }
67
89
  function parse(value) {
@@ -106,4 +128,4 @@ function parse(value) {
106
128
  }
107
129
  return cells;
108
130
  }
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"]}
131
+ //# 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:RIGHT |\n * ----------------------------------------------\n * | BOTTOMN:LEFT | BOTTOM:CENTER | BOTTOM:RIGHT |\n * -----------------------------------------------\n */\nimport { Rect } from \"@ngutil/style\"\n\nconst vertical = [\"top\", \"middle\", \"bottom\"] as const\nconst horizontal = [\"left\", \"center\", \"right\"] as const\n\nexport type L9Vertical = (typeof vertical)[number]\nexport type L9Horizontal = (typeof horizontal)[number]\nexport type L9CellName = `${L9Vertical}:${L9Horizontal}` | `${L9Horizontal}:${L9Vertical}`\nexport type L9RangeName = L9Vertical | L9Horizontal | L9CellName | `${L9CellName}-${L9CellName}`\nexport type L9Orient = \"horizontal\" | \"vertical\"\n\nexport const L9GridTopLeft = { row: 1, col: 1 }\n\nexport class L9Cell {\n    static coerce(value: L9CellName) {\n        const [v1, v2] = value.split(\":\")\n        const v = vertical.includes(v1 as any) ? v1 : v2\n        const h = horizontal.includes(v1 as any) ? v1 : v2\n\n        if (v === h) {\n            throw new Error(`Invalid cell value: ${value}`)\n        }\n\n        return new L9Cell(v as any, h as any)\n    }\n\n    constructor(\n        public readonly v: L9Vertical,\n        public readonly h: L9Horizontal\n    ) {}\n\n    intoGridArea(gridTopLeft: typeof L9GridTopLeft = L9GridTopLeft) {\n        return `${gridTopLeft.row + vertical.indexOf(this.v)}/${gridTopLeft.col + horizontal.indexOf(this.h)}`\n    }\n\n    isEq(other: L9Cell) {\n        return this.v === other.v && this.h === other.h\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    intoGridArea(gridTopLeft: typeof L9GridTopLeft = L9GridTopLeft) {\n        const start = this.cells[0]\n        const end = this.cells[this.cells.length - 1]\n        const endTopLeft = { row: gridTopLeft.row + 1, col: gridTopLeft.col + 1 }\n        return `${start.intoGridArea(gridTopLeft)}/${end.intoGridArea(endTopLeft)}`\n    }\n\n    intoRect(): Rect {\n        const start = this.cells[0]\n        const end = this.cells[this.cells.length - 1]\n        const x = horizontal.indexOf(start.h)\n        const y = vertical.indexOf(start.v)\n\n        return { x, y, width: horizontal.indexOf(end.h) - x + 1, height: vertical.indexOf(end.v) - y + 1 }\n    }\n\n    #determineOrient(): L9Orient {\n        const rect = this.intoRect()\n        if (rect.width === rect.height) {\n            // corner\n            if (rect.x === rect.y) {\n                return \"vertical\"\n            }\n\n            if (rect.x === 0 || rect.x === 2) {\n                return \"vertical\"\n            } else if (rect.y === 0 || rect.y === 2) {\n                return \"horizontal\"\n            }\n        } else if (rect.width > rect.height) {\n            return \"horizontal\"\n        } else if (rect.height > rect.width) {\n            return \"vertical\"\n        }\n\n        return \"vertical\"\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"]}