angular-toolbox 1.2.2 → 1.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -1
- package/esm2022/lib/component/index.mjs +2 -1
- package/esm2022/lib/component/layout/border-layout/border-layout.component.mjs +125 -0
- package/esm2022/lib/component/layout/border-layout/util/border-layout-bounds-manager.mjs +211 -0
- package/esm2022/lib/component/layout/border-layout/util/border-layout-renderer.mjs +212 -0
- package/esm2022/lib/component/layout/border-layout/util/resize-method.mjs +9 -0
- package/esm2022/lib/component/layout/border-layout-container/border-layout-container.component.mjs +143 -0
- package/esm2022/lib/component/layout/index.mjs +3 -0
- package/esm2022/lib/core/impl/version/version.impl.mjs +3 -4
- package/esm2022/lib/model/business/ui/index.mjs +2 -1
- package/esm2022/lib/model/business/ui/layout/index.mjs +5 -0
- package/esm2022/lib/model/business/ui/layout/layout-constraints.mjs +9 -0
- package/esm2022/lib/model/business/ui/layout/layout-drag-event-type.enum.mjs +26 -0
- package/esm2022/lib/model/business/ui/layout/layout-drag.event.mjs +23 -0
- package/esm2022/lib/model/business/ui/layout/layout-region.mjs +35 -0
- package/esm2022/lib/model/business/ui/window/window-init.mjs +1 -1
- package/esm2022/lib/model/service/version/angular-toolbox-version.service.mjs +4 -4
- package/esm2022/lib/util/index.mjs +2 -1
- package/esm2022/lib/util/version.util.mjs +44 -0
- package/esm2022/lib/util/window/window-features-builder.mjs +13 -1
- package/fesm2022/angular-toolbox.mjs +2524 -1703
- package/fesm2022/angular-toolbox.mjs.map +1 -1
- package/lib/component/index.d.ts +1 -0
- package/lib/component/layout/border-layout/border-layout.component.d.ts +79 -0
- package/lib/component/layout/border-layout/util/border-layout-bounds-manager.d.ts +131 -0
- package/lib/component/layout/border-layout/util/border-layout-renderer.d.ts +87 -0
- package/lib/component/layout/border-layout/util/resize-method.d.ts +13 -0
- package/lib/component/layout/border-layout-container/border-layout-container.component.d.ts +104 -0
- package/lib/component/layout/index.d.ts +2 -0
- package/lib/model/business/ui/index.d.ts +1 -0
- package/lib/model/business/ui/layout/index.d.ts +4 -0
- package/lib/model/business/ui/layout/layout-constraints.d.ts +33 -0
- package/lib/model/business/ui/layout/layout-drag-event-type.enum.d.ts +24 -0
- package/lib/model/business/ui/layout/layout-drag.event.d.ts +32 -0
- package/lib/model/business/ui/layout/layout-region.d.ts +36 -0
- package/lib/model/business/ui/window/window-init.d.ts +5 -0
- package/lib/util/index.d.ts +1 -0
- package/lib/util/version.util.d.ts +30 -0
- package/lib/util/window/window-features-builder.d.ts +4 -0
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
|
|
3
3
|
A library that provides useful tools for Angular apps development.
|
|
4
4
|
|
|
5
|
-
[]()
|
|
6
6
|
|
|
7
7
|
## License
|
|
8
8
|
|
|
@@ -6,4 +6,5 @@
|
|
|
6
6
|
* found in the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license
|
|
7
7
|
*/
|
|
8
8
|
export * from './angular-toolbox-logo/angular-toolbox-logo.component';
|
|
9
|
-
|
|
9
|
+
export * from './layout';
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hbmd1bGFyLXRvb2xib3gvc3JjL2xpYi9jb21wb25lbnQvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7OztHQU1HO0FBRUgsY0FBYyx1REFBdUQsQ0FBQztBQUN0RSxjQUFjLFVBQVUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxyXG4gKiBAbGljZW5zZVxyXG4gKiBDb3B5cmlnaHQgUGFzY2FsIEVDSEVNQU5OLiBBbGwgUmlnaHRzIFJlc2VydmVkLlxyXG4gKlxyXG4gKiBVc2Ugb2YgdGhpcyBzb3VyY2UgY29kZSBpcyBnb3Zlcm5lZCBieSBhbiBNSVQtc3R5bGUgbGljZW5zZSB0aGF0IGNhbiBiZVxyXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vcGFzY2FsZWNoZW1hbm4uY29tL2FuZ3VsYXItdG9vbGJveC9yZXNvdXJjZXMvbGljZW5zZVxyXG4gKi9cclxuXHJcbmV4cG9ydCAqIGZyb20gJy4vYW5ndWxhci10b29sYm94LWxvZ28vYW5ndWxhci10b29sYm94LWxvZ28uY29tcG9uZW50JztcclxuZXhwb3J0ICogZnJvbSAnLi9sYXlvdXQnO1xyXG4iXX0=
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Pascal ECHEMANN. All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Use of this source code is governed by an MIT-style license that can be found in
|
|
6
|
+
* the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license
|
|
7
|
+
*/
|
|
8
|
+
import { Component, ContentChildren, ViewChild, HostListener, EventEmitter, Output } from '@angular/core';
|
|
9
|
+
import { BorderLayoutContainer } from '../border-layout-container/border-layout-container.component';
|
|
10
|
+
import { LayoutDragEventType } from '../../../model';
|
|
11
|
+
import { BorderLayoutRenderer } from './util/border-layout-renderer';
|
|
12
|
+
import { IdentifiableComponent } from '../../../core';
|
|
13
|
+
import * as i0 from "@angular/core";
|
|
14
|
+
import * as i1 from "../../../model";
|
|
15
|
+
/**
|
|
16
|
+
* A border layout lays out a container, arranging and resizing its components to fit in five regions: north, south, east, west, and center.
|
|
17
|
+
* Each region is defined by a `BorderLayoutContainer` instance, and is identified by a corresponding constant: `NORTH`, `SOUTH`, `EAST`, `WEST`, and `CENTER`.
|
|
18
|
+
*/
|
|
19
|
+
export class BorderLayout extends IdentifiableComponent {
|
|
20
|
+
/**
|
|
21
|
+
* @private
|
|
22
|
+
*/
|
|
23
|
+
onResize() {
|
|
24
|
+
this.paint();
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* @private
|
|
28
|
+
*/
|
|
29
|
+
set __containers__(containers) {
|
|
30
|
+
this.renderer.addContainers(containers);
|
|
31
|
+
}
|
|
32
|
+
;
|
|
33
|
+
/**
|
|
34
|
+
* @private
|
|
35
|
+
*/
|
|
36
|
+
constructor(subscribeSvc) {
|
|
37
|
+
super();
|
|
38
|
+
this.subscribeSvc = subscribeSvc;
|
|
39
|
+
/**
|
|
40
|
+
* Emits events each time the user starts dragging a region handle.
|
|
41
|
+
*/
|
|
42
|
+
this.dragStart = new EventEmitter(false);
|
|
43
|
+
/**
|
|
44
|
+
* Emits events each time the user stops dragging a region handle.
|
|
45
|
+
*/
|
|
46
|
+
this.dragStop = new EventEmitter(false);
|
|
47
|
+
/**
|
|
48
|
+
* Emits events each time the user is dragging a region handle.
|
|
49
|
+
*/
|
|
50
|
+
this.dragging = new EventEmitter(false);
|
|
51
|
+
this.renderer = new BorderLayoutRenderer(subscribeSvc);
|
|
52
|
+
subscribeSvc.register(this, this.renderer.userAction.subscribe((event) => {
|
|
53
|
+
event.layout = this;
|
|
54
|
+
if (event.type === LayoutDragEventType.DRAGGING) {
|
|
55
|
+
this.dragging.emit(event);
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
if (event.type === LayoutDragEventType.DRAG_START) {
|
|
59
|
+
this.dragStart.emit(event);
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
this.dragStop.emit(event);
|
|
63
|
+
}));
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* @private
|
|
67
|
+
* For test purpose only.
|
|
68
|
+
*/
|
|
69
|
+
getRenderer() {
|
|
70
|
+
return this.renderer;
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* @private
|
|
74
|
+
*/
|
|
75
|
+
ngOnDestroy() {
|
|
76
|
+
this.subscribeSvc.clearAll(this);
|
|
77
|
+
this.renderer.destroy();
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* @private
|
|
81
|
+
*/
|
|
82
|
+
ngAfterViewInit() {
|
|
83
|
+
this.renderer.setLayoutContainer(this.layoutContainer.nativeElement);
|
|
84
|
+
this.paint();
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Forces the container to be redrawn.
|
|
88
|
+
*/
|
|
89
|
+
paint() {
|
|
90
|
+
this.renderer.paint();
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Resizes the specified region.
|
|
94
|
+
*
|
|
95
|
+
* @param region The region to resize.
|
|
96
|
+
* @param size The new size of the region to resize.
|
|
97
|
+
*
|
|
98
|
+
* @returns `true` whether the specified region has been resized; `false` otherwise.
|
|
99
|
+
*/
|
|
100
|
+
resizeRegion(region, size) {
|
|
101
|
+
this.renderer.resizeRegion(region, size);
|
|
102
|
+
}
|
|
103
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.3", ngImport: i0, type: BorderLayout, deps: [{ token: i1.SubscriptionService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
104
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.0.3", type: BorderLayout, isStandalone: true, selector: "atx-border-layout", outputs: { dragStart: "dragStart", dragStop: "dragStop", dragging: "dragging" }, host: { listeners: { "window:resize": "onResize()" } }, queries: [{ propertyName: "__containers__", predicate: BorderLayoutContainer }], viewQueries: [{ propertyName: "layoutContainer", first: true, predicate: ["atxLayoutContainer"], descendants: true }], usesInheritance: true, ngImport: i0, template: "<!--\n * LICENSE\n * Copyright Pascal ECHEMANN. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license\n-->\n\n<div #atxLayoutContainer class=\"atx-border-layout\">\n <ng-content></ng-content>\n</div>", styles: [".atx-border-layout{position:relative;height:100%;width:100%;-webkit-user-select:none;user-select:none;overflow:hidden}.atx-border-layout,.atx-border-layout *{box-sizing:border-box}.atx-border-layout .content{height:100%;width:100%;overflow:auto}.atx-border-layout .drag{position:absolute;border:2px solid black}:host ::ng-deep .atx-handle{position:absolute}:host ::ng-deep .atx-handle:hover,:host ::ng-deep .atx-handle.atx-handle-selected{background-color:var(--atx-handle-color, #469afa)}:host ::ng-deep .north{position:absolute;z-index:10;top:0;left:0;right:0}:host ::ng-deep .north .atx-handle{left:0;right:0;bottom:0;height:4px;cursor:ns-resize}:host ::ng-deep .west{position:absolute;z-index:5;left:0;height:unset!important}:host ::ng-deep .west .atx-handle{right:0;bottom:0;top:0;width:4px;cursor:ew-resize}:host ::ng-deep .east{position:absolute;z-index:5;right:0;height:unset!important}:host ::ng-deep .east .atx-handle{left:0;bottom:0;top:0;width:4px;cursor:ew-resize}:host ::ng-deep .south{position:absolute;z-index:10;bottom:0;left:0;right:0}:host ::ng-deep .south .atx-handle{left:0;right:0;top:0;height:4px;cursor:ns-resize}:host ::ng-deep .center{position:absolute}\n/**\n * @license\n * Copyright Pascal ECHEMANN. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be found in\n * the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license\n */\n"] }); }
|
|
105
|
+
}
|
|
106
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.3", ngImport: i0, type: BorderLayout, decorators: [{
|
|
107
|
+
type: Component,
|
|
108
|
+
args: [{ selector: 'atx-border-layout', standalone: true, template: "<!--\n * LICENSE\n * Copyright Pascal ECHEMANN. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license\n-->\n\n<div #atxLayoutContainer class=\"atx-border-layout\">\n <ng-content></ng-content>\n</div>", styles: [".atx-border-layout{position:relative;height:100%;width:100%;-webkit-user-select:none;user-select:none;overflow:hidden}.atx-border-layout,.atx-border-layout *{box-sizing:border-box}.atx-border-layout .content{height:100%;width:100%;overflow:auto}.atx-border-layout .drag{position:absolute;border:2px solid black}:host ::ng-deep .atx-handle{position:absolute}:host ::ng-deep .atx-handle:hover,:host ::ng-deep .atx-handle.atx-handle-selected{background-color:var(--atx-handle-color, #469afa)}:host ::ng-deep .north{position:absolute;z-index:10;top:0;left:0;right:0}:host ::ng-deep .north .atx-handle{left:0;right:0;bottom:0;height:4px;cursor:ns-resize}:host ::ng-deep .west{position:absolute;z-index:5;left:0;height:unset!important}:host ::ng-deep .west .atx-handle{right:0;bottom:0;top:0;width:4px;cursor:ew-resize}:host ::ng-deep .east{position:absolute;z-index:5;right:0;height:unset!important}:host ::ng-deep .east .atx-handle{left:0;bottom:0;top:0;width:4px;cursor:ew-resize}:host ::ng-deep .south{position:absolute;z-index:10;bottom:0;left:0;right:0}:host ::ng-deep .south .atx-handle{left:0;right:0;top:0;height:4px;cursor:ns-resize}:host ::ng-deep .center{position:absolute}\n/**\n * @license\n * Copyright Pascal ECHEMANN. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be found in\n * the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license\n */\n"] }]
|
|
109
|
+
}], ctorParameters: () => [{ type: i1.SubscriptionService }], propDecorators: { dragStart: [{
|
|
110
|
+
type: Output
|
|
111
|
+
}], dragStop: [{
|
|
112
|
+
type: Output
|
|
113
|
+
}], dragging: [{
|
|
114
|
+
type: Output
|
|
115
|
+
}], onResize: [{
|
|
116
|
+
type: HostListener,
|
|
117
|
+
args: ["window:resize"]
|
|
118
|
+
}], layoutContainer: [{
|
|
119
|
+
type: ViewChild,
|
|
120
|
+
args: ["atxLayoutContainer"]
|
|
121
|
+
}], __containers__: [{
|
|
122
|
+
type: ContentChildren,
|
|
123
|
+
args: [BorderLayoutContainer]
|
|
124
|
+
}] } });
|
|
125
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"border-layout.component.js","sourceRoot":"","sources":["../../../../../../../projects/angular-toolbox/src/lib/component/layout/border-layout/border-layout.component.ts","../../../../../../../projects/angular-toolbox/src/lib/component/layout/border-layout/border-layout.component.html"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAiB,SAAS,EAAE,eAAe,EAAyB,SAAS,EAAE,YAAY,EAAa,YAAY,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AACnK,OAAO,EAAE,qBAAqB,EAAE,MAAM,8DAA8D,CAAC;AACrG,OAAO,EAAmB,mBAAmB,EAAuD,MAAM,gBAAgB,CAAC;AAC3H,OAAO,EAAE,oBAAoB,EAAE,MAAM,+BAA+B,CAAC;AACrE,OAAO,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAC;;;AAEtD;;;GAGG;AAOH,MAAM,OAAO,YAAa,SAAQ,qBAAqB;IAoBrD;;OAEG;IAEK,QAAQ;QACd,IAAI,CAAC,KAAK,EAAE,CAAC;IACf,CAAC;IAaD;;OAEG;IACH,IACY,cAAc,CAAC,UAA4C;QACrE,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;IAC1C,CAAC;IAAA,CAAC;IAEF;;OAEG;IACH,YAAoB,YAAiC;QACnD,KAAK,EAAE,CAAC;QADU,iBAAY,GAAZ,YAAY,CAAqB;QAhDrD;;WAEG;QAEa,cAAS,GAAkC,IAAI,YAAY,CAAC,KAAK,CAAC,CAAC;QAEnF;;WAEG;QAEa,aAAQ,GAAkC,IAAI,YAAY,CAAC,KAAK,CAAC,CAAC;QAElF;;WAEG;QAEa,aAAQ,GAAkC,IAAI,YAAY,CAAC,KAAK,CAAC,CAAC;QAkChF,IAAI,CAAC,QAAQ,GAAG,IAAI,oBAAoB,CAAC,YAAY,CAAC,CAAC;QACvD,YAAY,CAAC,QAAQ,CAAC,IAAI,EACxB,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,KAAsB,EAAC,EAAE;YAC3D,KAAK,CAAC,MAAM,GAAG,IAAI,CAAC;YACpB,IAAI,KAAK,CAAC,IAAI,KAAK,mBAAmB,CAAC,QAAQ,EAAE,CAAC;gBAChD,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC1B,OAAO;YACT,CAAC;YACD,IAAI,KAAK,CAAC,IAAI,KAAK,mBAAmB,CAAC,UAAU,EAAE,CAAC;gBAClD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC3B,OAAO;YACT,CAAC;YACD,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5B,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED;;;OAGG;IACI,WAAW;QAChB,OAAO,IAAI,CAAC,QAAQ,CAAC;IACvB,CAAC;IAED;;OAEG;IACI,WAAW;QAChB,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACjC,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC;IAC1B,CAAC;IAED;;OAEG;IACI,eAAe;QACpB,IAAI,CAAC,QAAQ,CAAC,kBAAkB,CAAC,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,CAAC;QACrE,IAAI,CAAC,KAAK,EAAE,CAAC;IACf,CAAC;IAED;;OAEG;IACI,KAAK;QACV,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC;IACxB,CAAC;IAED;;;;;;;OAOG;IACI,YAAY,CAAC,MAAuC,EAAE,IAAY;QACvE,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,MAAsB,EAAE,IAAI,CAAC,CAAC;IAC3D,CAAC;8GA9GU,YAAY;kGAAZ,YAAY,qPA0CN,qBAAqB,2KClExC,yVAUM;;2FDcO,YAAY;kBANxB,SAAS;+BACE,mBAAmB,cAGjB,IAAI;wFAQA,SAAS;sBADxB,MAAM;gBAOS,QAAQ;sBADvB,MAAM;gBAOS,QAAQ;sBADvB,MAAM;gBAOC,QAAQ;sBADf,YAAY;uBAAC,eAAe;gBASrB,eAAe;sBADtB,SAAS;uBAAC,oBAAoB;gBAYnB,cAAc;sBADzB,eAAe;uBAAC,qBAAqB","sourcesContent":["/**\n * @license\n * Copyright Pascal ECHEMANN. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be found in\n * the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license\n */\n\nimport { AfterViewInit, Component, ContentChildren, OnDestroy, ElementRef, ViewChild, HostListener, QueryList, EventEmitter, output, Output } from '@angular/core';\nimport { BorderLayoutContainer } from '../border-layout-container/border-layout-container.component';\nimport { LayoutDragEvent, LayoutDragEventType, LayoutRegion, LayoutRegionType, SubscriptionService } from '../../../model';\nimport { BorderLayoutRenderer } from './util/border-layout-renderer';\nimport { IdentifiableComponent } from '../../../core';\n\n/**\n * A border layout lays out a container, arranging and resizing its components to fit in five regions: north, south, east, west, and center.\n * Each region is defined by a `BorderLayoutContainer` instance, and is identified by a corresponding constant: `NORTH`, `SOUTH`, `EAST`, `WEST`, and `CENTER`.\n */\n@Component({\n  selector: 'atx-border-layout',\n  templateUrl: './border-layout.component.html',\n  styleUrls: ['./border-layout.component.scss'],\n  standalone: true\n})\nexport class BorderLayout extends IdentifiableComponent implements AfterViewInit, OnDestroy {\n\n  /**\n   * Emits events each time the user starts dragging a region handle.\n   */\n  @Output()\n  public readonly dragStart: EventEmitter<LayoutDragEvent> = new EventEmitter(false);\n\n  /**\n   * Emits events each time the user stops dragging a region handle.\n   */\n  @Output()\n  public readonly dragStop: EventEmitter<LayoutDragEvent> = new EventEmitter(false);\n\n  /**\n   * Emits events each time the user is dragging a region handle.\n   */\n  @Output()\n  public readonly dragging: EventEmitter<LayoutDragEvent> = new EventEmitter(false);\n  \n  /**\n   * @private\n   */\n  @HostListener(\"window:resize\")\n  private onResize(): void {\n    this.paint();\n  }\n\n  /**\n   * @private\n   */\n  @ViewChild(\"atxLayoutContainer\")\n  private layoutContainer!: ElementRef<HTMLDivElement>;\n\n  /**\n   * @private\n   */\n  private renderer: BorderLayoutRenderer;\n\n  /**\n   * @private\n   */\n  @ContentChildren(BorderLayoutContainer)\n  private set __containers__(containers: QueryList<BorderLayoutContainer>) {\n    this.renderer.addContainers(containers);\n  };\n\n  /**\n   * @private\n   */\n  constructor(private subscribeSvc: SubscriptionService) {\n    super();\n    this.renderer = new BorderLayoutRenderer(subscribeSvc);\n    subscribeSvc.register(this,\n      this.renderer.userAction.subscribe((event: LayoutDragEvent)=> {\n        event.layout = this;\n        if (event.type === LayoutDragEventType.DRAGGING) {\n          this.dragging.emit(event);\n          return;\n        }\n        if (event.type === LayoutDragEventType.DRAG_START) {\n          this.dragStart.emit(event);\n          return;\n        }\n        this.dragStop.emit(event);\n      })\n    );\n  }\n\n  /**\n   * @private\n   * For test purpose only.\n   */\n  public getRenderer(): BorderLayoutRenderer {\n    return this.renderer;\n  }\n\n  /**\n   * @private\n   */\n  public ngOnDestroy(): void {\n    this.subscribeSvc.clearAll(this);\n    this.renderer.destroy();\n  }\n\n  /**\n   * @private\n   */\n  public ngAfterViewInit(): void {\n    this.renderer.setLayoutContainer(this.layoutContainer.nativeElement);\n    this.paint();\n  }\n\n  /**\n   * Forces the container to be redrawn.\n   */\n  public paint(): void {\n    this.renderer.paint();\n  }\n\n  /**\n   * Resizes the specified region.\n   * \n   * @param region The region to resize.\n   * @param size The new size of the region to resize.\n   * \n   * @returns `true` whether the specified region has been resized; `false` otherwise.\n   */\n  public resizeRegion(region: LayoutRegion | LayoutRegionType, size: number): void {\n    this.renderer.resizeRegion(region as LayoutRegion, size);\n  }\n}\n","<!--\n * LICENSE\n * Copyright Pascal ECHEMANN. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license\n-->\n\n<div #atxLayoutContainer class=\"atx-border-layout\">\n    <ng-content></ng-content>\n</div>"]}
|
|
@@ -0,0 +1,211 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Pascal ECHEMANN. All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Use of this source code is governed by an MIT-style license that can be found in
|
|
6
|
+
* the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license
|
|
7
|
+
*/
|
|
8
|
+
import { LayoutRegion, } from '../../../../model';
|
|
9
|
+
/**
|
|
10
|
+
* @private
|
|
11
|
+
* A utility class that computes the size of all border layout containers.
|
|
12
|
+
*/
|
|
13
|
+
export class BorderLayoutBoundsManager {
|
|
14
|
+
constructor() {
|
|
15
|
+
/**
|
|
16
|
+
* @private
|
|
17
|
+
* The container top position when the user starts to drag the handle.
|
|
18
|
+
*/
|
|
19
|
+
this.topPos = 0;
|
|
20
|
+
/**
|
|
21
|
+
* @private
|
|
22
|
+
* The container left position when the user starts to drag the handle.
|
|
23
|
+
*/
|
|
24
|
+
this.leftPos = 0;
|
|
25
|
+
/**
|
|
26
|
+
* @private
|
|
27
|
+
* The bounds of the center container.
|
|
28
|
+
*/
|
|
29
|
+
this.bounds = new DOMRect();
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Makes this object elligible for garbage collection.
|
|
33
|
+
*/
|
|
34
|
+
destroy() {
|
|
35
|
+
this.bounds = null;
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Returns a rectangle that represents the bounds currently computed by this object.
|
|
39
|
+
*
|
|
40
|
+
* @returns The bounds currently computed by this object.
|
|
41
|
+
*/
|
|
42
|
+
getBounds() {
|
|
43
|
+
return this.bounds;
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Sets the top-left hand corner position before each bounds computaion.
|
|
47
|
+
*
|
|
48
|
+
* @param x The x position of the main container, in pixels, before computaion.
|
|
49
|
+
* @param y The y position of the main container, in pixels, before computaion.
|
|
50
|
+
*/
|
|
51
|
+
setOrigin(x, y) {
|
|
52
|
+
this.leftPos = x;
|
|
53
|
+
this.topPos = y;
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* @private
|
|
57
|
+
* For test purpose only.
|
|
58
|
+
*/
|
|
59
|
+
getOrigin() {
|
|
60
|
+
return { x: this.leftPos, y: this.topPos };
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Computes the size of a "north" container.
|
|
64
|
+
*
|
|
65
|
+
* @param event The mouse event that triggered the computation of the container size.
|
|
66
|
+
* @param width The width of the main container.
|
|
67
|
+
* @param height The height of the main container.
|
|
68
|
+
* @param minSize The minimum size of the container for which to compute the size.
|
|
69
|
+
* @param maxSize The maximum size of the container for which to compute the size.
|
|
70
|
+
*
|
|
71
|
+
* @returns The size of a "north" container.
|
|
72
|
+
*/
|
|
73
|
+
northResize(event, width, height, minSize, maxSize) {
|
|
74
|
+
const size = event.clientY - this.topPos;
|
|
75
|
+
if (maxSize && size > maxSize)
|
|
76
|
+
return this.setY(maxSize);
|
|
77
|
+
if (minSize && size < minSize)
|
|
78
|
+
return this.setY(minSize);
|
|
79
|
+
return this.setY(size);
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* @private
|
|
83
|
+
*/
|
|
84
|
+
setY(y) {
|
|
85
|
+
this.bounds.y = y;
|
|
86
|
+
return y;
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Computes the size of a "south" container.
|
|
90
|
+
*
|
|
91
|
+
* @param event The mouse event that triggered the computation of the container size.
|
|
92
|
+
* @param width The width of the main container.
|
|
93
|
+
* @param height The height of the main container.
|
|
94
|
+
* @param minSize The minimum size of the container for which to compute the size.
|
|
95
|
+
* @param maxSize The maximum size of the container for which to compute the size.
|
|
96
|
+
*
|
|
97
|
+
* @returns The size of a "south" container.
|
|
98
|
+
*/
|
|
99
|
+
southResize(event, width, height, minSize, maxSize) {
|
|
100
|
+
const size = height - (event.clientY - this.topPos);
|
|
101
|
+
if (maxSize && size > maxSize)
|
|
102
|
+
return this.setHeight(maxSize);
|
|
103
|
+
if (minSize && size < minSize)
|
|
104
|
+
return this.setHeight(minSize);
|
|
105
|
+
return this.setHeight(size);
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* @private
|
|
109
|
+
*/
|
|
110
|
+
setHeight(height) {
|
|
111
|
+
this.bounds.height = height;
|
|
112
|
+
return height;
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Computes the size of a "west" container.
|
|
116
|
+
*
|
|
117
|
+
* @param event The mouse event that triggered the computation of the container size.
|
|
118
|
+
* @param width The width of the main container.
|
|
119
|
+
* @param height The height of the main container.
|
|
120
|
+
* @param minSize The minimum size of the container for which to compute the size.
|
|
121
|
+
* @param maxSize The maximum size of the container for which to compute the size.
|
|
122
|
+
*
|
|
123
|
+
* @returns The size of a "west" container.
|
|
124
|
+
*/
|
|
125
|
+
westResize(event, width, height, minSize, maxSize) {
|
|
126
|
+
const size = event.clientX - this.leftPos;
|
|
127
|
+
if (maxSize && size > maxSize)
|
|
128
|
+
return this.setX(maxSize);
|
|
129
|
+
if (minSize && size < minSize)
|
|
130
|
+
return this.setX(minSize);
|
|
131
|
+
return this.setX(size);
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* @private
|
|
135
|
+
*/
|
|
136
|
+
setX(x) {
|
|
137
|
+
this.bounds.x = x;
|
|
138
|
+
return x;
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Computes the size of a "east" container.
|
|
142
|
+
*
|
|
143
|
+
* @param event The mouse event that triggered the computation of the container size.
|
|
144
|
+
* @param width The width of the main container.
|
|
145
|
+
* @param height The height of the main container.
|
|
146
|
+
* @param minSize The minimum size of the container for which to compute the size.
|
|
147
|
+
* @param maxSize The maximum size of the container for which to compute the size.
|
|
148
|
+
*
|
|
149
|
+
* @returns The size of a "east" container.
|
|
150
|
+
*/
|
|
151
|
+
eastResize(event, width, height, minSize, maxSize) {
|
|
152
|
+
const size = width - (event.clientX - this.leftPos);
|
|
153
|
+
if (maxSize && size > maxSize)
|
|
154
|
+
return this.setWidth(maxSize);
|
|
155
|
+
if (minSize && size < minSize)
|
|
156
|
+
return this.setWidth(minSize);
|
|
157
|
+
return this.setWidth(size);
|
|
158
|
+
}
|
|
159
|
+
/**
|
|
160
|
+
* @private
|
|
161
|
+
*/
|
|
162
|
+
setWidth(width) {
|
|
163
|
+
this.bounds.width = width;
|
|
164
|
+
return width;
|
|
165
|
+
}
|
|
166
|
+
/**
|
|
167
|
+
* Invoked after each `BorderLayoutContainer` registration to initialize the center container bounds.
|
|
168
|
+
*
|
|
169
|
+
* @param container A `BorderLayoutContainer` instance added to the main container.
|
|
170
|
+
*/
|
|
171
|
+
initBounds(container) {
|
|
172
|
+
const r = container.constraints.region;
|
|
173
|
+
if (r === LayoutRegion.NORTH) {
|
|
174
|
+
this.bounds.y = container.getSize();
|
|
175
|
+
return;
|
|
176
|
+
}
|
|
177
|
+
if (r === LayoutRegion.SOUTH) {
|
|
178
|
+
this.bounds.height = container.getSize();
|
|
179
|
+
return;
|
|
180
|
+
}
|
|
181
|
+
if (r === LayoutRegion.WEST) {
|
|
182
|
+
this.bounds.x = container.getSize();
|
|
183
|
+
return;
|
|
184
|
+
}
|
|
185
|
+
if (r === LayoutRegion.EAST)
|
|
186
|
+
this.bounds.width = container.getSize();
|
|
187
|
+
}
|
|
188
|
+
/**
|
|
189
|
+
* Returns the method used to compute the size of a container, depending on the specified region.
|
|
190
|
+
*
|
|
191
|
+
* @param region The region associated with the container for which to compute the size.
|
|
192
|
+
*
|
|
193
|
+
* @returns The method used to compute the size of a container.
|
|
194
|
+
*/
|
|
195
|
+
getResizeMethod(region) {
|
|
196
|
+
let method = this.southResize;
|
|
197
|
+
switch (region) {
|
|
198
|
+
case LayoutRegion.NORTH:
|
|
199
|
+
method = this.northResize;
|
|
200
|
+
break;
|
|
201
|
+
case LayoutRegion.WEST:
|
|
202
|
+
method = this.westResize;
|
|
203
|
+
break;
|
|
204
|
+
case LayoutRegion.EAST:
|
|
205
|
+
method = this.eastResize;
|
|
206
|
+
break;
|
|
207
|
+
}
|
|
208
|
+
return method.bind(this);
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"border-layout-bounds-manager.js","sourceRoot":"","sources":["../../../../../../../../projects/angular-toolbox/src/lib/component/layout/border-layout/util/border-layout-bounds-manager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Pascal ECHEMANN. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be found in\n * the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license\n */\n\nimport { BorderLayoutContainer } from '../../border-layout-container/border-layout-container.component';\nimport { Destroyable, LayoutRegion, } from '../../../../model';\nimport { ResizeMethod } from './resize-method';\n\n/**\n * @private\n * A utility class that computes the size of all border layout containers.\n */\nexport class BorderLayoutBoundsManager implements Destroyable {\n\n  /**\n   * @private\n   * The container top position when the user starts to drag the handle.\n   */\n  private topPos: number = 0;\n\n  /**\n   * @private\n   * The container left position when the user starts to drag the handle.\n   */\n  private leftPos: number = 0;\n\n  /**\n   * @private\n   * The bounds of the center container.\n   */\n  private bounds: DOMRect = new DOMRect();\n\n  /**\n   * Makes this object elligible for garbage collection.\n   */\n  public destroy(): void {\n    this.bounds = null as any;\n  }\n\n  /**\n   * Returns a rectangle that represents the bounds currently computed by this object.\n   * \n   * @returns The bounds currently computed by this object.\n   */\n  public getBounds(): DOMRect {\n    return this.bounds;\n  }\n\n  /**\n   * Sets the top-left hand corner position before each bounds computaion.\n   * \n   * @param x The x position of the main container, in pixels, before computaion.\n   * @param y The y position of the main container, in pixels, before computaion.\n   */\n  public setOrigin(x: number, y: number): void {\n    this.leftPos = x;\n    this.topPos = y;\n  }\n\n  /**\n   * @private\n   * For test purpose only.\n   */\n  public getOrigin(): any {\n    return { x: this.leftPos, y: this.topPos };\n  }\n\n  /**\n   * Computes the size of a \"north\" container.\n   * \n   * @param event The mouse event that triggered the computation of the container size.\n   * @param width The width of the main container.\n   * @param height The height of the main container.\n   * @param minSize The minimum size of the container for which to compute the size.\n   * @param maxSize The maximum size of the container for which to compute the size.\n   * \n   * @returns The size of a \"north\" container.\n   */\n  public northResize(event: MouseEvent, width: number, height: number, minSize: number | undefined, maxSize: number | undefined): number {\n    const size: number = event.clientY - this.topPos;\n    if (maxSize && size > maxSize) return this.setY(maxSize);\n    if (minSize && size < minSize) return this.setY(minSize);\n    return this.setY(size);\n  }\n\n  /**\n   * @private\n   */\n  private setY(y: number): number {\n    this.bounds.y = y;\n    return y;\n  }\n\n  /**\n   * Computes the size of a \"south\" container.\n   * \n   * @param event The mouse event that triggered the computation of the container size.\n   * @param width The width of the main container.\n   * @param height The height of the main container.\n   * @param minSize The minimum size of the container for which to compute the size.\n   * @param maxSize The maximum size of the container for which to compute the size.\n   * \n   * @returns The size of a \"south\" container.\n   */\n  public southResize(event: MouseEvent, width: number, height: number, minSize: number | undefined, maxSize: number | undefined): number {\n    const size: number = height - (event.clientY - this.topPos);\n    if (maxSize && size > maxSize) return this.setHeight(maxSize);\n    if (minSize && size < minSize) return this.setHeight(minSize);\n    return this.setHeight(size);\n  }\n\n  /**\n   * @private\n   */\n  private setHeight(height: number): number {\n    this.bounds.height = height;\n    return height;\n  }\n\n  /**\n   * Computes the size of a \"west\" container.\n   * \n   * @param event The mouse event that triggered the computation of the container size.\n   * @param width The width of the main container.\n   * @param height The height of the main container.\n   * @param minSize The minimum size of the container for which to compute the size.\n   * @param maxSize The maximum size of the container for which to compute the size.\n   * \n   * @returns The size of a \"west\" container.\n   */\n  public westResize(event: MouseEvent, width: number, height: number, minSize: number | undefined, maxSize: number | undefined): number {\n    const size: number = event.clientX - this.leftPos;\n    if (maxSize && size > maxSize) return this.setX(maxSize);\n    if (minSize && size < minSize) return this.setX(minSize);\n    return this.setX(size);\n  }\n\n  /**\n   * @private\n   */\n  private setX(x: number): number {\n    this.bounds.x = x;\n    return x;\n  }\n\n  /**\n   * Computes the size of a \"east\" container.\n   * \n   * @param event The mouse event that triggered the computation of the container size.\n   * @param width The width of the main container.\n   * @param height The height of the main container.\n   * @param minSize The minimum size of the container for which to compute the size.\n   * @param maxSize The maximum size of the container for which to compute the size.\n   * \n   * @returns The size of a \"east\" container.\n   */\n  public eastResize(event: MouseEvent, width: number, height: number, minSize: number | undefined, maxSize: number | undefined): number {\n    const size: number = width - (event.clientX - this.leftPos);\n    if (maxSize && size > maxSize) return this.setWidth(maxSize);\n    if (minSize && size < minSize) return this.setWidth(minSize);\n    return this.setWidth(size);\n  }\n\n  /**\n   * @private\n   */\n  private setWidth(width: number): number {\n    this.bounds.width = width;\n    return width;\n  }\n\n  /**\n   * Invoked after each `BorderLayoutContainer` registration to initialize the center container bounds.\n   * \n   * @param container A `BorderLayoutContainer` instance added to the main container. \n   */\n  public initBounds(container: BorderLayoutContainer): void {\n    const r: LayoutRegion = container.constraints.region as LayoutRegion;\n    if (r === LayoutRegion.NORTH) {\n      this.bounds.y = container.getSize();\n      return;\n    }\n    if (r === LayoutRegion.SOUTH) {\n      this.bounds.height = container.getSize();\n      return;\n    }\n    if (r === LayoutRegion.WEST) {\n      this.bounds.x = container.getSize();\n      return;\n    }\n    if (r === LayoutRegion.EAST) this.bounds.width = container.getSize();\n  }\n  \n  /**\n   * Returns the method used to compute the size of a container, depending on the specified region.\n   * \n   * @param region The region associated with the container for which to compute the size.\n   * \n   * @returns The method used to compute the size of a container.\n   */\n  public getResizeMethod(region: LayoutRegion): ResizeMethod {\n    let method: ResizeMethod = this.southResize;\n    switch (region) {\n      case LayoutRegion.NORTH: method = this.northResize; break;\n      case LayoutRegion.WEST: method = this.westResize; break;\n      case LayoutRegion.EAST: method = this.eastResize; break;\n    }\n    return method.bind(this) as any;\n  }\n}\n"]}
|
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Pascal ECHEMANN. All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Use of this source code is governed by an MIT-style license that can be found in
|
|
6
|
+
* the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license
|
|
7
|
+
*/
|
|
8
|
+
import { EventEmitter } from '@angular/core';
|
|
9
|
+
import { EMPTY_STRING } from '../../../../util';
|
|
10
|
+
import { LayoutDragEvent, LayoutDragEventType, LayoutRegion } from '../../../../model';
|
|
11
|
+
import { IdentifiableComponent } from '../../../../core';
|
|
12
|
+
import { BorderLayoutBoundsManager } from './border-layout-bounds-manager';
|
|
13
|
+
/**
|
|
14
|
+
* @private
|
|
15
|
+
*/
|
|
16
|
+
const MOUSEMOVE = "mousemove";
|
|
17
|
+
/**
|
|
18
|
+
* @private
|
|
19
|
+
*/
|
|
20
|
+
const MOUSEUP = "mouseup";
|
|
21
|
+
/**
|
|
22
|
+
* @private
|
|
23
|
+
*/
|
|
24
|
+
const REGION_ERR_MSG = "A container with the same identifier has already been registered: ";
|
|
25
|
+
/**
|
|
26
|
+
* @private
|
|
27
|
+
*/
|
|
28
|
+
const LAYOUT_ERR_MSG = "No layout container has been registered.";
|
|
29
|
+
/**
|
|
30
|
+
* @private
|
|
31
|
+
* A controller object responsible for handling user actions on a `BorderLayout` container.
|
|
32
|
+
*/
|
|
33
|
+
export class BorderLayoutRenderer extends IdentifiableComponent {
|
|
34
|
+
/**
|
|
35
|
+
* @private
|
|
36
|
+
*/
|
|
37
|
+
constructor(subscribeSvc) {
|
|
38
|
+
super();
|
|
39
|
+
this.subscribeSvc = subscribeSvc;
|
|
40
|
+
/**
|
|
41
|
+
* Emits events each time the user starts, or stops dragging handle.
|
|
42
|
+
*/
|
|
43
|
+
this.userAction = new EventEmitter(false);
|
|
44
|
+
/**
|
|
45
|
+
* @private
|
|
46
|
+
*/
|
|
47
|
+
this.containerList = [];
|
|
48
|
+
this.boundsManager = new BorderLayoutBoundsManager();
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Add the list `BorderLayoutContainer` objects associated with the main container to this controller.
|
|
52
|
+
*
|
|
53
|
+
* @param containers the list `BorderLayoutContainer` objects associated with the main container.
|
|
54
|
+
*/
|
|
55
|
+
addContainers(containers) {
|
|
56
|
+
let regionValidator = EMPTY_STRING;
|
|
57
|
+
containers.forEach(container => {
|
|
58
|
+
const constraints = container.constraints;
|
|
59
|
+
const r = constraints.region;
|
|
60
|
+
if (regionValidator.indexOf(r) !== -1)
|
|
61
|
+
throw new SyntaxError(REGION_ERR_MSG + r);
|
|
62
|
+
regionValidator += r;
|
|
63
|
+
if (constraints.resizable) {
|
|
64
|
+
this.subscribeSvc.register(this, container.resizeStart.subscribe(container => this.resizeEnter(container)));
|
|
65
|
+
}
|
|
66
|
+
this.boundsManager.initBounds(container);
|
|
67
|
+
if (r === LayoutRegion.NORTH || r === LayoutRegion.SOUTH)
|
|
68
|
+
return;
|
|
69
|
+
this.containerList.push(container);
|
|
70
|
+
});
|
|
71
|
+
}
|
|
72
|
+
;
|
|
73
|
+
/**
|
|
74
|
+
* Sets the reference to the HTML container associated with the main container.
|
|
75
|
+
*
|
|
76
|
+
* @param lytContainer The reference to the HTML container associated with the main container.
|
|
77
|
+
*/
|
|
78
|
+
setLayoutContainer(lytContainer) {
|
|
79
|
+
this.lytContainerElm = lytContainer;
|
|
80
|
+
}
|
|
81
|
+
;
|
|
82
|
+
/**
|
|
83
|
+
* @private
|
|
84
|
+
* For test purpose only.
|
|
85
|
+
* Returns the reference to the internal `BorderLayoutBoundsManager` instance.
|
|
86
|
+
*/
|
|
87
|
+
getBoundsManager() {
|
|
88
|
+
return this.boundsManager;
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
* Makes this object elligible for garbage collection.
|
|
92
|
+
*/
|
|
93
|
+
destroy() {
|
|
94
|
+
this.subscribeSvc.clearAll(this);
|
|
95
|
+
this.containerList.length = 0;
|
|
96
|
+
this.containerList = null;
|
|
97
|
+
this.lytContainerElm = null;
|
|
98
|
+
this.boundsManager.destroy();
|
|
99
|
+
this.boundsManager = null;
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Forces the layout of all `BorderLayoutContainer` objects associated with the main container.
|
|
103
|
+
*/
|
|
104
|
+
paint() {
|
|
105
|
+
this.checkLytContainer();
|
|
106
|
+
this.render(this.lytContainerElm.offsetWidth);
|
|
107
|
+
}
|
|
108
|
+
/**
|
|
109
|
+
* Resizes the specified region of the associated container.
|
|
110
|
+
*
|
|
111
|
+
* @param region The region to resize.
|
|
112
|
+
* @param size The new size of the region to resize.
|
|
113
|
+
*
|
|
114
|
+
* @returns `true` whether the specified region has been resized; `false` otherwise.
|
|
115
|
+
*/
|
|
116
|
+
resizeRegion(region, size) {
|
|
117
|
+
if (region === LayoutRegion.CENTER)
|
|
118
|
+
return false;
|
|
119
|
+
const container = this.containerList.find((c) => {
|
|
120
|
+
return c.constraints.region === region;
|
|
121
|
+
});
|
|
122
|
+
if (!container)
|
|
123
|
+
return false;
|
|
124
|
+
container.setSize(size);
|
|
125
|
+
this.boundsManager.initBounds(container);
|
|
126
|
+
this.paint();
|
|
127
|
+
return true;
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* @private
|
|
131
|
+
*/
|
|
132
|
+
resizeEnter(container) {
|
|
133
|
+
this.checkLytContainer();
|
|
134
|
+
const lytNativeElm = this.lytContainerElm;
|
|
135
|
+
const width = lytNativeElm.offsetWidth;
|
|
136
|
+
const height = lytNativeElm.offsetHeight;
|
|
137
|
+
const bounds = lytNativeElm.getBoundingClientRect();
|
|
138
|
+
const constraints = container.constraints;
|
|
139
|
+
const region = constraints.region;
|
|
140
|
+
const minSize = constraints.minSize;
|
|
141
|
+
const maxSize = constraints.maxSize;
|
|
142
|
+
let size = 0;
|
|
143
|
+
this.boundsManager.setOrigin(bounds.x, bounds.y);
|
|
144
|
+
let resizeMethod = this.boundsManager.getResizeMethod(region);
|
|
145
|
+
container.selected = true;
|
|
146
|
+
const onMoveHandler = (event) => {
|
|
147
|
+
event.preventDefault();
|
|
148
|
+
event.stopPropagation();
|
|
149
|
+
size = resizeMethod(event, width, height, minSize, maxSize);
|
|
150
|
+
container.setSize(size);
|
|
151
|
+
this.render(width);
|
|
152
|
+
this.fireEvent(container, LayoutDragEventType.DRAGGING);
|
|
153
|
+
};
|
|
154
|
+
const onStopHandler = (event) => {
|
|
155
|
+
event.preventDefault();
|
|
156
|
+
event.stopPropagation();
|
|
157
|
+
lytNativeElm.removeEventListener(MOUSEMOVE, onMoveHandler);
|
|
158
|
+
lytNativeElm.removeEventListener(MOUSEUP, onStopHandler);
|
|
159
|
+
container.selected = false;
|
|
160
|
+
size = resizeMethod(event, width, height, minSize, maxSize);
|
|
161
|
+
container.setSize(size);
|
|
162
|
+
this.render(width);
|
|
163
|
+
this.fireEvent(container, LayoutDragEventType.DRAG_STOP);
|
|
164
|
+
};
|
|
165
|
+
lytNativeElm.addEventListener(MOUSEMOVE, onMoveHandler);
|
|
166
|
+
lytNativeElm.addEventListener(MOUSEUP, onStopHandler);
|
|
167
|
+
this.fireEvent(container, LayoutDragEventType.DRAG_START);
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* @private
|
|
171
|
+
*/
|
|
172
|
+
fireEvent(target, type) {
|
|
173
|
+
const evt = new LayoutDragEvent(target, type);
|
|
174
|
+
this.userAction.emit(evt);
|
|
175
|
+
}
|
|
176
|
+
/**
|
|
177
|
+
* @private
|
|
178
|
+
*/
|
|
179
|
+
render(width) {
|
|
180
|
+
const bounds = this.boundsManager.getBounds();
|
|
181
|
+
const top = bounds.top;
|
|
182
|
+
const bottom = bounds.bottom;
|
|
183
|
+
const left = bounds.left;
|
|
184
|
+
const right = bounds.right;
|
|
185
|
+
this.containerList.forEach((cont) => {
|
|
186
|
+
const r = cont.constraints.region;
|
|
187
|
+
if (r === LayoutRegion.WEST) {
|
|
188
|
+
cont.setTopPos(top);
|
|
189
|
+
cont.setRightPos(width - left);
|
|
190
|
+
cont.setBottomPos(bottom - top);
|
|
191
|
+
return;
|
|
192
|
+
}
|
|
193
|
+
if (r === LayoutRegion.CENTER) {
|
|
194
|
+
cont.setTopPos(top);
|
|
195
|
+
cont.setLeftPos(left);
|
|
196
|
+
cont.setRightPos(right - left);
|
|
197
|
+
cont.setBottomPos(bottom - top);
|
|
198
|
+
return;
|
|
199
|
+
}
|
|
200
|
+
if (r === LayoutRegion.EAST) {
|
|
201
|
+
cont.setTopPos(top);
|
|
202
|
+
cont.setLeftPos(width - (right - left));
|
|
203
|
+
cont.setBottomPos(bottom - top);
|
|
204
|
+
}
|
|
205
|
+
});
|
|
206
|
+
}
|
|
207
|
+
checkLytContainer() {
|
|
208
|
+
if (!this.lytContainerElm)
|
|
209
|
+
throw new ReferenceError(LAYOUT_ERR_MSG);
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"border-layout-renderer.js","sourceRoot":"","sources":["../../../../../../../../projects/angular-toolbox/src/lib/component/layout/border-layout/util/border-layout-renderer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Pascal ECHEMANN. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be found in\n * the LICENSE file at https://pascalechemann.com/angular-toolbox/resources/license\n */\n\nimport { EventEmitter, QueryList } from '@angular/core';\nimport { BorderLayoutContainer } from '../../border-layout-container/border-layout-container.component';\nimport { EMPTY_STRING } from '../../../../util';\nimport { Destroyable, LayoutConstraints, LayoutDragEvent, LayoutDragEventType, LayoutRegion, SubscriptionService } from '../../../../model';\nimport { IdentifiableComponent } from '../../../../core';\nimport { BorderLayoutBoundsManager } from './border-layout-bounds-manager';\nimport { ResizeMethod } from './resize-method';\n\n/**\n * @private\n */\nconst MOUSEMOVE: any = \"mousemove\";\n\n/**\n * @private\n */\nconst MOUSEUP: any = \"mouseup\";\n\n/**\n * @private\n */\nconst REGION_ERR_MSG: string = \"A container with the same identifier has already been registered: \";\n\n/**\n * @private\n */\nconst LAYOUT_ERR_MSG: string = \"No layout container has been registered.\";\n\n/**\n * @private\n * A controller object responsible for handling user actions on a `BorderLayout` container.\n */\nexport class BorderLayoutRenderer extends IdentifiableComponent implements Destroyable {\n\n  /**\n   * Emits events each time the user starts, or stops dragging handle.\n   */\n  public readonly userAction: EventEmitter<LayoutDragEvent> = new EventEmitter(false);\n\n  /**\n   * @private\n   */\n  private lytContainerElm!: HTMLDivElement;\n  \n  /**\n   * @private\n   */\n  private containerList: BorderLayoutContainer[] = [];\n  \n  /**\n   * @private\n   */\n  private boundsManager: BorderLayoutBoundsManager; \n\n  /**\n   * @private\n   */\n  constructor(private subscribeSvc: SubscriptionService) {\n    super();\n    this.boundsManager = new BorderLayoutBoundsManager();\n  }\n\n  /**\n   * Add the list `BorderLayoutContainer` objects associated with the main container to this controller.\n   * \n   * @param containers the list `BorderLayoutContainer` objects associated with the main container.\n   */\n  public addContainers(containers: QueryList<BorderLayoutContainer>): void {\n    let regionValidator: string = EMPTY_STRING;\n    containers.forEach(container => {\n      const constraints: LayoutConstraints = container.constraints;\n      const r: string = constraints.region as string;\n      if (regionValidator.indexOf(r) !== -1) throw new SyntaxError(REGION_ERR_MSG + r);\n      regionValidator += r;\n      if (constraints.resizable) {\n        this.subscribeSvc.register(this,\n          container.resizeStart.subscribe(container=> this.resizeEnter(container))\n        );\n      }\n      this.boundsManager.initBounds(container);\n      if (r === LayoutRegion.NORTH || r === LayoutRegion.SOUTH) return;\n      this.containerList.push(container);\n    });\n  };\n\n  /**\n   * Sets the reference to the HTML container associated with the main container.\n   * \n   * @param lytContainer The reference to the HTML container associated with the main container.\n   */\n  public setLayoutContainer(lytContainer: HTMLDivElement): void {\n    this.lytContainerElm = lytContainer;\n  };\n\n  /**\n   * @private\n   * For test purpose only.\n   * Returns the reference to the internal `BorderLayoutBoundsManager` instance.\n   */\n  public getBoundsManager(): BorderLayoutBoundsManager {\n    return this.boundsManager; \n  }\n\n  /**\n   * Makes this object elligible for garbage collection.\n   */\n  public destroy(): void {\n    this.subscribeSvc.clearAll(this);\n    this.containerList.length = 0;\n    this.containerList = null as any;\n    this.lytContainerElm = null as any;\n    this.boundsManager.destroy();\n    this.boundsManager = null as any;\n  }\n\n  /**\n   * Forces the layout of all `BorderLayoutContainer` objects associated with the main container.\n   */\n  public paint(): void {\n    this.checkLytContainer();\n    this.render(this.lytContainerElm.offsetWidth);\n  }\n\n  /**\n   * Resizes the specified region of the associated container.\n   * \n   * @param region The region to resize.\n   * @param size The new size of the region to resize.\n   * \n   * @returns `true` whether the specified region has been resized; `false` otherwise.\n   */\n  public resizeRegion(region: LayoutRegion, size: number): boolean {\n    if (region === LayoutRegion.CENTER) return false;\n    const container: BorderLayoutContainer | undefined = this.containerList.find((c: BorderLayoutContainer)=> {\n      return c.constraints.region === region;\n    });\n    if (!container) return false;\n    container.setSize(size);\n    this.boundsManager.initBounds(container);\n    this.paint();\n    return true;\n  }\n\n  /**\n   * @private\n   */\n  private resizeEnter(container: BorderLayoutContainer): void {\n    this.checkLytContainer();\n    const lytNativeElm: HTMLDivElement = this.lytContainerElm;\n    const width: number = lytNativeElm.offsetWidth;\n    const height: number = lytNativeElm.offsetHeight;\n    const bounds: DOMRect = lytNativeElm.getBoundingClientRect();\n    const constraints: LayoutConstraints = container.constraints;\n    const region: LayoutRegion = constraints.region as LayoutRegion;\n    const minSize: number | undefined = constraints.minSize;\n    const maxSize: number | undefined = constraints.maxSize;\n    let size: number = 0;\n    this.boundsManager.setOrigin(bounds.x, bounds.y);\n    let resizeMethod: ResizeMethod = this.boundsManager.getResizeMethod(region);\n    container.selected = true;\n    const onMoveHandler = (event: MouseEvent)=> {\n      event.preventDefault();\n      event.stopPropagation();\n      size = resizeMethod(event, width, height, minSize, maxSize);\n      container.setSize(size);\n      this.render(width);\n      this.fireEvent(container, LayoutDragEventType.DRAGGING);\n    };\n    const onStopHandler = (event: MouseEvent)=> {\n      event.preventDefault();\n      event.stopPropagation();\n      lytNativeElm.removeEventListener(MOUSEMOVE, onMoveHandler);\n      lytNativeElm.removeEventListener(MOUSEUP, onStopHandler);\n      container.selected = false;\n      size = resizeMethod(event, width, height, minSize, maxSize);\n      container.setSize(size);\n      this.render(width);\n      this.fireEvent(container, LayoutDragEventType.DRAG_STOP);\n    };\n    lytNativeElm.addEventListener(MOUSEMOVE, onMoveHandler);\n    lytNativeElm.addEventListener(MOUSEUP, onStopHandler);\n    this.fireEvent(container, LayoutDragEventType.DRAG_START);\n  }\n\n  /**\n   * @private\n   */\n  private fireEvent(target: BorderLayoutContainer, type: LayoutDragEventType): void {\n    const evt: LayoutDragEvent = new LayoutDragEvent(target, type);\n    this.userAction.emit(evt);\n  }\n\n  /**\n   * @private\n   */\n  private render(width: number): void {\n    const bounds: DOMRect = this.boundsManager.getBounds();\n    const top: number = bounds.top;\n    const bottom: number = bounds.bottom;\n    const left: number = bounds.left;\n    const right: number = bounds.right;\n    this.containerList.forEach((cont: BorderLayoutContainer) => {\n      const r: LayoutRegion = cont.constraints.region as LayoutRegion;\n      if (r === LayoutRegion.WEST) {\n        cont.setTopPos(top);\n        cont.setRightPos(width - left);\n        cont.setBottomPos(bottom - top);\n        return;\n      }\n      if (r === LayoutRegion.CENTER) {\n        cont.setTopPos(top);\n        cont.setLeftPos(left);\n        cont.setRightPos(right - left);\n        cont.setBottomPos(bottom - top);\n        return;\n      }\n      if (r === LayoutRegion.EAST) {\n        cont.setTopPos(top);\n        cont.setLeftPos(width - (right - left));\n        cont.setBottomPos(bottom - top);\n      }\n    });\n  }\n\n  private checkLytContainer(): void {\n    if (!this.lytContainerElm) throw new ReferenceError(LAYOUT_ERR_MSG);\n  }\n}\n"]}
|