concepto-user-controls 0.0.8 → 0.0.10
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/concepto-input/concepto-input.component.mjs +20 -0
- package/esm2022/lib/concepto-tree/components/tree-node/tree-node.component.mjs +301 -0
- package/esm2022/lib/concepto-tree/components/tree-node-checkbox/tree-node-checkbox.component.mjs +90 -0
- package/esm2022/lib/concepto-tree/components/tree-node-content/tree-node-content.component.mjs +65 -0
- package/esm2022/lib/concepto-tree/components/tree-node-expander/tree-node-expander.component.mjs +74 -0
- package/esm2022/lib/concepto-tree/components/tree-root/tree-root.component.mjs +230 -0
- package/esm2022/lib/concepto-tree/components/tree-viewport/tree-viewport.component.mjs +216 -0
- package/esm2022/lib/concepto-tree/concepto-tree.component.mjs +214 -13
- package/esm2022/lib/concepto-tree/core/models/tree-events.model.mjs +2 -0
- package/esm2022/lib/concepto-tree/core/models/tree-node.model.mjs +102 -0
- package/esm2022/lib/concepto-tree/core/models/tree-options.model.mjs +24 -0
- package/esm2022/lib/concepto-tree/core/models/tree.model.mjs +313 -0
- package/esm2022/lib/concepto-tree/core/services/tree-drag-drop.service.mjs +27 -0
- package/esm2022/lib/concepto-tree/directives/tree-drag.directive.mjs +69 -0
- package/esm2022/lib/concepto-tree/directives/tree-drop.directive.mjs +124 -0
- package/esm2022/lib/concepto-tree/directives/tree-node-template.directive.mjs +19 -0
- package/esm2022/public-api.mjs +2 -1
- package/fesm2022/concepto-user-controls.mjs +1837 -13
- package/fesm2022/concepto-user-controls.mjs.map +1 -1
- package/lib/concepto-input/concepto-input.component.d.ts +8 -0
- package/lib/concepto-tree/components/tree-node/tree-node.component.d.ts +19 -0
- package/lib/concepto-tree/components/tree-node-checkbox/tree-node-checkbox.component.d.ts +17 -0
- package/lib/concepto-tree/components/tree-node-content/tree-node-content.component.d.ts +18 -0
- package/lib/concepto-tree/components/tree-node-expander/tree-node-expander.component.d.ts +12 -0
- package/lib/concepto-tree/components/tree-root/tree-root.component.d.ts +35 -0
- package/lib/concepto-tree/components/tree-viewport/tree-viewport.component.d.ts +33 -0
- package/lib/concepto-tree/concepto-tree.component.d.ts +32 -1
- package/lib/concepto-tree/core/models/tree-events.model.d.ts +13 -0
- package/lib/concepto-tree/core/models/tree-node.model.d.ts +39 -0
- package/lib/concepto-tree/core/models/tree-options.model.d.ts +28 -0
- package/lib/concepto-tree/core/models/tree.model.d.ts +54 -0
- package/lib/concepto-tree/core/services/tree-drag-drop.service.d.ts +11 -0
- package/lib/concepto-tree/directives/tree-drag.directive.d.ts +16 -0
- package/lib/concepto-tree/directives/tree-drop.directive.d.ts +25 -0
- package/lib/concepto-tree/directives/tree-node-template.directive.d.ts +8 -0
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { Component, Input } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export class ConceptoInputComponent {
|
|
4
|
+
labelText = 'Date';
|
|
5
|
+
name = 'date-input';
|
|
6
|
+
type = 'text';
|
|
7
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ConceptoInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
8
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ConceptoInputComponent, isStandalone: true, selector: "concepto-input", inputs: { labelText: "labelText", name: "name", type: "type" }, ngImport: i0, template: "<div class=\"field\">\r\n @if(labelText !== '') {\r\n <label [for]=\"name\" class=\"label\">{{labelText}}</label>\r\n }\r\n @if(type === 'text') {\r\n <input [name]=\"name\" [id]=\"name\" type=\"text\" class=\"text-input\" />\r\n } @else if(type === 'date') {\r\n <input [name]=\"name\" [id]=\"name\" type=\"date\" class=\"date-input\" />\r\n } @else if(type === 'dropdown') {\r\n <select [name]=\"name\" [id]=\"name\" class=\"dropdown-input\">\r\n <option value=\"\" disabled selected>Select an option</option>\r\n </select>\r\n }\r\n</div>", styles: [".field{display:flex;flex-direction:column;align-items:flex-start;gap:10px;width:100%;margin:20px 0}.label{font-family:var(--font);font-style:normal;font-weight:400;font-size:12px;line-height:15px;color:var(--primary-text-color)}.date-input,.text-input,.dropdown-input{box-sizing:border-box;width:100%;height:35px;padding:10px;border:1px solid var(--secondary-color);border-radius:5px;font-family:var(--font);font-style:normal;font-weight:400;font-size:12px;line-height:15px;color:var(--primary-text-color);background-color:var( --primary-background-color )}.dropdown-input{appearance:none;-webkit-appearance:none;-moz-appearance:none;background-image:url('data:image/svg+xml;utf8,<svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 9L12 15L18 9\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke=\"currentColor\"/></svg>');background-repeat:no-repeat;background-position:right 10px center;background-size:18px}.dropdown-input:open{background-image:url('data:image/svg+xml;utf8,<svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 15L12 9L18 15\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke=\"currentColor\"/></svg>')}.date-input:focus,.text-input:focus,.dropdown-input:focus{outline:none;border-color:#0e0e12}.date-input::-webkit-calendar-picker-indicator{cursor:pointer}\n"] });
|
|
9
|
+
}
|
|
10
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ConceptoInputComponent, decorators: [{
|
|
11
|
+
type: Component,
|
|
12
|
+
args: [{ selector: 'concepto-input', standalone: true, imports: [], template: "<div class=\"field\">\r\n @if(labelText !== '') {\r\n <label [for]=\"name\" class=\"label\">{{labelText}}</label>\r\n }\r\n @if(type === 'text') {\r\n <input [name]=\"name\" [id]=\"name\" type=\"text\" class=\"text-input\" />\r\n } @else if(type === 'date') {\r\n <input [name]=\"name\" [id]=\"name\" type=\"date\" class=\"date-input\" />\r\n } @else if(type === 'dropdown') {\r\n <select [name]=\"name\" [id]=\"name\" class=\"dropdown-input\">\r\n <option value=\"\" disabled selected>Select an option</option>\r\n </select>\r\n }\r\n</div>", styles: [".field{display:flex;flex-direction:column;align-items:flex-start;gap:10px;width:100%;margin:20px 0}.label{font-family:var(--font);font-style:normal;font-weight:400;font-size:12px;line-height:15px;color:var(--primary-text-color)}.date-input,.text-input,.dropdown-input{box-sizing:border-box;width:100%;height:35px;padding:10px;border:1px solid var(--secondary-color);border-radius:5px;font-family:var(--font);font-style:normal;font-weight:400;font-size:12px;line-height:15px;color:var(--primary-text-color);background-color:var( --primary-background-color )}.dropdown-input{appearance:none;-webkit-appearance:none;-moz-appearance:none;background-image:url('data:image/svg+xml;utf8,<svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 9L12 15L18 9\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke=\"currentColor\"/></svg>');background-repeat:no-repeat;background-position:right 10px center;background-size:18px}.dropdown-input:open{background-image:url('data:image/svg+xml;utf8,<svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M6 15L12 9L18 15\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke=\"currentColor\"/></svg>')}.date-input:focus,.text-input:focus,.dropdown-input:focus{outline:none;border-color:#0e0e12}.date-input::-webkit-calendar-picker-indicator{cursor:pointer}\n"] }]
|
|
13
|
+
}], propDecorators: { labelText: [{
|
|
14
|
+
type: Input
|
|
15
|
+
}], name: [{
|
|
16
|
+
type: Input
|
|
17
|
+
}], type: [{
|
|
18
|
+
type: Input
|
|
19
|
+
}] } });
|
|
20
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uY2VwdG8taW5wdXQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY29uY2VwdG8tdXNlci1jb250cm9scy9zcmMvbGliL2NvbmNlcHRvLWlucHV0L2NvbmNlcHRvLWlucHV0LmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbmNlcHRvLXVzZXItY29udHJvbHMvc3JjL2xpYi9jb25jZXB0by1pbnB1dC9jb25jZXB0by1pbnB1dC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxNQUFNLGVBQWUsQ0FBQzs7QUFTakQsTUFBTSxPQUFPLHNCQUFzQjtJQUN4QixTQUFTLEdBQVcsTUFBTSxDQUFDO0lBQzNCLElBQUksR0FBVyxZQUFZLENBQUM7SUFDNUIsSUFBSSxHQUFXLE1BQU0sQ0FBQzt3R0FIcEIsc0JBQXNCOzRGQUF0QixzQkFBc0IsMElDVG5DLHVrQkFhTTs7NEZESk8sc0JBQXNCO2tCQVBsQyxTQUFTOytCQUNFLGdCQUFnQixjQUNkLElBQUksV0FDUCxFQUFFOzhCQUtGLFNBQVM7c0JBQWpCLEtBQUs7Z0JBQ0csSUFBSTtzQkFBWixLQUFLO2dCQUNHLElBQUk7c0JBQVosS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuXHJcbkBDb21wb25lbnQoe1xyXG4gIHNlbGVjdG9yOiAnY29uY2VwdG8taW5wdXQnLFxyXG4gIHN0YW5kYWxvbmU6IHRydWUsXHJcbiAgaW1wb3J0czogW10sXHJcbiAgdGVtcGxhdGVVcmw6ICcuL2NvbmNlcHRvLWlucHV0LmNvbXBvbmVudC5odG1sJyxcclxuICBzdHlsZVVybDogJy4vY29uY2VwdG8taW5wdXQuY29tcG9uZW50LmNzcydcclxufSlcclxuZXhwb3J0IGNsYXNzIENvbmNlcHRvSW5wdXRDb21wb25lbnQge1xyXG4gIEBJbnB1dCgpIGxhYmVsVGV4dDogc3RyaW5nID0gJ0RhdGUnO1xyXG4gIEBJbnB1dCgpIG5hbWU6IHN0cmluZyA9ICdkYXRlLWlucHV0JztcclxuICBASW5wdXQoKSB0eXBlOiBzdHJpbmcgPSAndGV4dCc7XHJcbn1cclxuIiwiPGRpdiBjbGFzcz1cImZpZWxkXCI+XHJcbiAgICBAaWYobGFiZWxUZXh0ICE9PSAnJykge1xyXG4gICAgPGxhYmVsIFtmb3JdPVwibmFtZVwiIGNsYXNzPVwibGFiZWxcIj57e2xhYmVsVGV4dH19PC9sYWJlbD5cclxuICAgIH1cclxuICAgIEBpZih0eXBlID09PSAndGV4dCcpIHtcclxuICAgIDxpbnB1dCBbbmFtZV09XCJuYW1lXCIgW2lkXT1cIm5hbWVcIiB0eXBlPVwidGV4dFwiIGNsYXNzPVwidGV4dC1pbnB1dFwiIC8+XHJcbiAgICB9IEBlbHNlIGlmKHR5cGUgPT09ICdkYXRlJykge1xyXG4gICAgPGlucHV0IFtuYW1lXT1cIm5hbWVcIiBbaWRdPVwibmFtZVwiIHR5cGU9XCJkYXRlXCIgY2xhc3M9XCJkYXRlLWlucHV0XCIgLz5cclxuICAgIH0gQGVsc2UgaWYodHlwZSA9PT0gJ2Ryb3Bkb3duJykge1xyXG4gICAgPHNlbGVjdCBbbmFtZV09XCJuYW1lXCIgW2lkXT1cIm5hbWVcIiBjbGFzcz1cImRyb3Bkb3duLWlucHV0XCI+XHJcbiAgICAgICAgPG9wdGlvbiB2YWx1ZT1cIlwiIGRpc2FibGVkIHNlbGVjdGVkPlNlbGVjdCBhbiBvcHRpb248L29wdGlvbj5cclxuICAgIDwvc2VsZWN0PlxyXG4gICAgfVxyXG48L2Rpdj4iXX0=
|
|
@@ -0,0 +1,301 @@
|
|
|
1
|
+
// lib/components/tree-node/tree-node.component.ts
|
|
2
|
+
import { Component, Input, ChangeDetectionStrategy, computed, } from '@angular/core';
|
|
3
|
+
import { CommonModule } from '@angular/common';
|
|
4
|
+
import { trigger, state, style, transition, animate } from '@angular/animations';
|
|
5
|
+
import { TreeNodeExpanderComponent } from '../tree-node-expander/tree-node-expander.component';
|
|
6
|
+
import { TreeNodeCheckboxComponent } from '../tree-node-checkbox/tree-node-checkbox.component';
|
|
7
|
+
import { TreeNodeContentComponent } from '../tree-node-content/tree-node-content.component';
|
|
8
|
+
import { TreeDragDirective } from '../../directives/tree-drag.directive';
|
|
9
|
+
import { TreeDropDirective } from '../../directives/tree-drop.directive';
|
|
10
|
+
import * as i0 from "@angular/core";
|
|
11
|
+
export class TreeNodeComponent {
|
|
12
|
+
node;
|
|
13
|
+
treeModel;
|
|
14
|
+
options;
|
|
15
|
+
nodeTemplate;
|
|
16
|
+
paddingLeft = computed(() => {
|
|
17
|
+
return this.options.levelPadding || 40;
|
|
18
|
+
});
|
|
19
|
+
onNodeClick(event) {
|
|
20
|
+
const useMetaKey = this.options.useMetaKey !== false;
|
|
21
|
+
const multiSelect = this.options.multiSelect || false;
|
|
22
|
+
const isMetaKeyPressed = event.ctrlKey || event.metaKey;
|
|
23
|
+
const shouldMultiSelect = multiSelect && (!useMetaKey || isMetaKeyPressed);
|
|
24
|
+
this.treeModel.activateNode(this.node, shouldMultiSelect);
|
|
25
|
+
this.treeModel.focusNode(this.node);
|
|
26
|
+
}
|
|
27
|
+
onNodeDoubleClick(event) {
|
|
28
|
+
if (this.node.hasChildren) {
|
|
29
|
+
this.node.toggle();
|
|
30
|
+
this.treeModel.expandedNodeIds.update(ids => {
|
|
31
|
+
const newIds = new Set(ids);
|
|
32
|
+
if (this.node.isExpanded()) {
|
|
33
|
+
newIds.add(this.node.id);
|
|
34
|
+
}
|
|
35
|
+
else {
|
|
36
|
+
newIds.delete(this.node.id);
|
|
37
|
+
}
|
|
38
|
+
return newIds;
|
|
39
|
+
});
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
onNodeContextMenu(event) {
|
|
43
|
+
event.preventDefault();
|
|
44
|
+
// Emit context menu event
|
|
45
|
+
}
|
|
46
|
+
onToggleExpanded() {
|
|
47
|
+
if (this.node.hasChildren) {
|
|
48
|
+
this.node.toggle();
|
|
49
|
+
if (this.node.isExpanded()) {
|
|
50
|
+
// Load children if needed
|
|
51
|
+
if (this.options.getChildren && this.node.children.length === 0) {
|
|
52
|
+
this.treeModel.loadChildren(this.node);
|
|
53
|
+
}
|
|
54
|
+
this.treeModel.expandNode(this.node);
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
this.treeModel.collapseNode(this.node);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
onCheckboxChange(checked) {
|
|
62
|
+
this.treeModel.selectNode(this.node, this.options.multiSelect || false);
|
|
63
|
+
}
|
|
64
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeNodeComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
65
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: TreeNodeComponent, isStandalone: true, selector: "tree-node", inputs: { node: "node", treeModel: "treeModel", options: "options", nodeTemplate: "nodeTemplate" }, ngImport: i0, template: `
|
|
66
|
+
<div
|
|
67
|
+
class="tree-node"
|
|
68
|
+
[class.tree-node-expanded]="node.isExpanded()"
|
|
69
|
+
[class.tree-node-collapsed]="node.isCollapsed()"
|
|
70
|
+
[class.tree-node-active]="node.isActive()"
|
|
71
|
+
[class.tree-node-focused]="node.isFocused()"
|
|
72
|
+
[class.tree-node-leaf]="node.isLeaf()"
|
|
73
|
+
[class.tree-node-loading]="node.isLoading()"
|
|
74
|
+
[style.padding-left.px]="paddingLeft()">
|
|
75
|
+
|
|
76
|
+
<div class="tree-node-wrapper"
|
|
77
|
+
[treeDrag]="node"
|
|
78
|
+
[treeModel]="treeModel"
|
|
79
|
+
[options]="options">
|
|
80
|
+
|
|
81
|
+
<!-- Drop zone before node -->
|
|
82
|
+
<div class="tree-node-drop-slot tree-node-drop-slot-before"
|
|
83
|
+
[treeDrop]="node"
|
|
84
|
+
[treeModel]="treeModel"
|
|
85
|
+
[dropPosition]="'before'"
|
|
86
|
+
[options]="options">
|
|
87
|
+
</div>
|
|
88
|
+
|
|
89
|
+
<!-- Node content wrapper -->
|
|
90
|
+
<div class="tree-node-content-wrapper"
|
|
91
|
+
(click)="onNodeClick($event)"
|
|
92
|
+
(dblclick)="onNodeDoubleClick($event)"
|
|
93
|
+
(contextmenu)="onNodeContextMenu($event)">
|
|
94
|
+
|
|
95
|
+
<!-- Expander -->
|
|
96
|
+
@if (node.hasChildren) {
|
|
97
|
+
<tree-node-expander
|
|
98
|
+
[node]="node"
|
|
99
|
+
[treeModel]="treeModel"
|
|
100
|
+
(toggle)="onToggleExpanded()">
|
|
101
|
+
</tree-node-expander>
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
<!-- Checkbox -->
|
|
105
|
+
@if (options.useCheckbox) {
|
|
106
|
+
<tree-node-checkbox
|
|
107
|
+
[node]="node"
|
|
108
|
+
[treeModel]="treeModel"
|
|
109
|
+
[useTriState]="options.useTriState ?? false"
|
|
110
|
+
(change)="onCheckboxChange($event)">
|
|
111
|
+
</tree-node-checkbox>
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
<!-- Content -->
|
|
115
|
+
<tree-node-content
|
|
116
|
+
[node]="node"
|
|
117
|
+
[treeModel]="treeModel"
|
|
118
|
+
[template]="nodeTemplate"
|
|
119
|
+
[displayField]="options.displayField || 'name'">
|
|
120
|
+
</tree-node-content>
|
|
121
|
+
</div>
|
|
122
|
+
|
|
123
|
+
<!-- Drop zone after node -->
|
|
124
|
+
<div class="tree-node-drop-slot tree-node-drop-slot-after"
|
|
125
|
+
[treeDrop]="node"
|
|
126
|
+
[treeModel]="treeModel"
|
|
127
|
+
[dropPosition]="'after'"
|
|
128
|
+
[options]="options">
|
|
129
|
+
</div>
|
|
130
|
+
</div>
|
|
131
|
+
|
|
132
|
+
<!-- Children container -->
|
|
133
|
+
@if (node.isExpanded() && node.children.length > 0) {
|
|
134
|
+
<div class="tree-node-children"
|
|
135
|
+
[@expandCollapse]="node.isExpanded() ? 'expanded' : 'collapsed'">
|
|
136
|
+
@for (child of node.children; track child.id) {
|
|
137
|
+
<tree-node
|
|
138
|
+
[node]="child"
|
|
139
|
+
[treeModel]="treeModel"
|
|
140
|
+
[options]="options"
|
|
141
|
+
[nodeTemplate]="nodeTemplate">
|
|
142
|
+
</tree-node>
|
|
143
|
+
}
|
|
144
|
+
</div>
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
<!-- Loading indicator -->
|
|
148
|
+
@if (node.isLoading()) {
|
|
149
|
+
<div class="tree-node-loading">
|
|
150
|
+
<span class="loading-spinner"></span>
|
|
151
|
+
Loading...
|
|
152
|
+
</div>
|
|
153
|
+
}
|
|
154
|
+
</div>
|
|
155
|
+
`, isInline: true, styles: [".tree-node{position:relative}.tree-node-wrapper{position:relative;display:flex;align-items:center;min-height:22px}.tree-node-content-wrapper{display:flex;align-items:center;flex:1;cursor:pointer;border-radius:3px;padding:2px 4px;transition:background-color .15s}.tree-node-content-wrapper:hover{background-color:#f5f5f5}.tree-node-active .tree-node-content-wrapper{background-color:#e3f2fd}.tree-node-focused .tree-node-content-wrapper{outline:2px solid #2196f3;outline-offset:-2px}.tree-node-children{overflow:hidden}.tree-node-drop-slot{position:absolute;left:0;right:0;height:2px;z-index:10}.tree-node-drop-slot-before{top:-1px}.tree-node-drop-slot-after{bottom:-1px}.tree-node-loading{padding:8px 24px;display:flex;align-items:center;gap:8px;color:#666;font-size:14px}.loading-spinner{display:inline-block;width:16px;height:16px;border:2px solid #f3f3f3;border-top:2px solid #2196f3;border-radius:50%;animation:spin 1s linear infinite}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"], dependencies: [{ kind: "component", type: TreeNodeComponent, selector: "tree-node", inputs: ["node", "treeModel", "options", "nodeTemplate"] }, { kind: "ngmodule", type: CommonModule }, { kind: "component", type: TreeNodeExpanderComponent, selector: "tree-node-expander", inputs: ["node", "treeModel"], outputs: ["toggle"] }, { kind: "component", type: TreeNodeCheckboxComponent, selector: "tree-node-checkbox", inputs: ["node", "treeModel", "useTriState"], outputs: ["change"] }, { kind: "component", type: TreeNodeContentComponent, selector: "tree-node-content", inputs: ["node", "treeModel", "template", "displayField"] }, { kind: "directive", type: TreeDragDirective, selector: "[treeDrag]", inputs: ["treeDrag", "treeModel", "options"] }, { kind: "directive", type: TreeDropDirective, selector: "[treeDrop]", inputs: ["treeDrop", "treeModel", "options", "dropPosition"] }], animations: [
|
|
156
|
+
trigger('expandCollapse', [
|
|
157
|
+
state('collapsed', style({
|
|
158
|
+
height: '0',
|
|
159
|
+
overflow: 'hidden',
|
|
160
|
+
opacity: '0'
|
|
161
|
+
})),
|
|
162
|
+
state('expanded', style({
|
|
163
|
+
height: '*',
|
|
164
|
+
overflow: 'visible',
|
|
165
|
+
opacity: '1'
|
|
166
|
+
})),
|
|
167
|
+
transition('collapsed <=> expanded', [
|
|
168
|
+
animate('200ms ease-in-out')
|
|
169
|
+
])
|
|
170
|
+
])
|
|
171
|
+
], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
172
|
+
}
|
|
173
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeNodeComponent, decorators: [{
|
|
174
|
+
type: Component,
|
|
175
|
+
args: [{ selector: 'tree-node', standalone: true, imports: [
|
|
176
|
+
CommonModule,
|
|
177
|
+
TreeNodeExpanderComponent,
|
|
178
|
+
TreeNodeCheckboxComponent,
|
|
179
|
+
TreeNodeContentComponent,
|
|
180
|
+
TreeDragDirective,
|
|
181
|
+
TreeDropDirective,
|
|
182
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: `
|
|
183
|
+
<div
|
|
184
|
+
class="tree-node"
|
|
185
|
+
[class.tree-node-expanded]="node.isExpanded()"
|
|
186
|
+
[class.tree-node-collapsed]="node.isCollapsed()"
|
|
187
|
+
[class.tree-node-active]="node.isActive()"
|
|
188
|
+
[class.tree-node-focused]="node.isFocused()"
|
|
189
|
+
[class.tree-node-leaf]="node.isLeaf()"
|
|
190
|
+
[class.tree-node-loading]="node.isLoading()"
|
|
191
|
+
[style.padding-left.px]="paddingLeft()">
|
|
192
|
+
|
|
193
|
+
<div class="tree-node-wrapper"
|
|
194
|
+
[treeDrag]="node"
|
|
195
|
+
[treeModel]="treeModel"
|
|
196
|
+
[options]="options">
|
|
197
|
+
|
|
198
|
+
<!-- Drop zone before node -->
|
|
199
|
+
<div class="tree-node-drop-slot tree-node-drop-slot-before"
|
|
200
|
+
[treeDrop]="node"
|
|
201
|
+
[treeModel]="treeModel"
|
|
202
|
+
[dropPosition]="'before'"
|
|
203
|
+
[options]="options">
|
|
204
|
+
</div>
|
|
205
|
+
|
|
206
|
+
<!-- Node content wrapper -->
|
|
207
|
+
<div class="tree-node-content-wrapper"
|
|
208
|
+
(click)="onNodeClick($event)"
|
|
209
|
+
(dblclick)="onNodeDoubleClick($event)"
|
|
210
|
+
(contextmenu)="onNodeContextMenu($event)">
|
|
211
|
+
|
|
212
|
+
<!-- Expander -->
|
|
213
|
+
@if (node.hasChildren) {
|
|
214
|
+
<tree-node-expander
|
|
215
|
+
[node]="node"
|
|
216
|
+
[treeModel]="treeModel"
|
|
217
|
+
(toggle)="onToggleExpanded()">
|
|
218
|
+
</tree-node-expander>
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
<!-- Checkbox -->
|
|
222
|
+
@if (options.useCheckbox) {
|
|
223
|
+
<tree-node-checkbox
|
|
224
|
+
[node]="node"
|
|
225
|
+
[treeModel]="treeModel"
|
|
226
|
+
[useTriState]="options.useTriState ?? false"
|
|
227
|
+
(change)="onCheckboxChange($event)">
|
|
228
|
+
</tree-node-checkbox>
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
<!-- Content -->
|
|
232
|
+
<tree-node-content
|
|
233
|
+
[node]="node"
|
|
234
|
+
[treeModel]="treeModel"
|
|
235
|
+
[template]="nodeTemplate"
|
|
236
|
+
[displayField]="options.displayField || 'name'">
|
|
237
|
+
</tree-node-content>
|
|
238
|
+
</div>
|
|
239
|
+
|
|
240
|
+
<!-- Drop zone after node -->
|
|
241
|
+
<div class="tree-node-drop-slot tree-node-drop-slot-after"
|
|
242
|
+
[treeDrop]="node"
|
|
243
|
+
[treeModel]="treeModel"
|
|
244
|
+
[dropPosition]="'after'"
|
|
245
|
+
[options]="options">
|
|
246
|
+
</div>
|
|
247
|
+
</div>
|
|
248
|
+
|
|
249
|
+
<!-- Children container -->
|
|
250
|
+
@if (node.isExpanded() && node.children.length > 0) {
|
|
251
|
+
<div class="tree-node-children"
|
|
252
|
+
[@expandCollapse]="node.isExpanded() ? 'expanded' : 'collapsed'">
|
|
253
|
+
@for (child of node.children; track child.id) {
|
|
254
|
+
<tree-node
|
|
255
|
+
[node]="child"
|
|
256
|
+
[treeModel]="treeModel"
|
|
257
|
+
[options]="options"
|
|
258
|
+
[nodeTemplate]="nodeTemplate">
|
|
259
|
+
</tree-node>
|
|
260
|
+
}
|
|
261
|
+
</div>
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
<!-- Loading indicator -->
|
|
265
|
+
@if (node.isLoading()) {
|
|
266
|
+
<div class="tree-node-loading">
|
|
267
|
+
<span class="loading-spinner"></span>
|
|
268
|
+
Loading...
|
|
269
|
+
</div>
|
|
270
|
+
}
|
|
271
|
+
</div>
|
|
272
|
+
`, animations: [
|
|
273
|
+
trigger('expandCollapse', [
|
|
274
|
+
state('collapsed', style({
|
|
275
|
+
height: '0',
|
|
276
|
+
overflow: 'hidden',
|
|
277
|
+
opacity: '0'
|
|
278
|
+
})),
|
|
279
|
+
state('expanded', style({
|
|
280
|
+
height: '*',
|
|
281
|
+
overflow: 'visible',
|
|
282
|
+
opacity: '1'
|
|
283
|
+
})),
|
|
284
|
+
transition('collapsed <=> expanded', [
|
|
285
|
+
animate('200ms ease-in-out')
|
|
286
|
+
])
|
|
287
|
+
])
|
|
288
|
+
], styles: [".tree-node{position:relative}.tree-node-wrapper{position:relative;display:flex;align-items:center;min-height:22px}.tree-node-content-wrapper{display:flex;align-items:center;flex:1;cursor:pointer;border-radius:3px;padding:2px 4px;transition:background-color .15s}.tree-node-content-wrapper:hover{background-color:#f5f5f5}.tree-node-active .tree-node-content-wrapper{background-color:#e3f2fd}.tree-node-focused .tree-node-content-wrapper{outline:2px solid #2196f3;outline-offset:-2px}.tree-node-children{overflow:hidden}.tree-node-drop-slot{position:absolute;left:0;right:0;height:2px;z-index:10}.tree-node-drop-slot-before{top:-1px}.tree-node-drop-slot-after{bottom:-1px}.tree-node-loading{padding:8px 24px;display:flex;align-items:center;gap:8px;color:#666;font-size:14px}.loading-spinner{display:inline-block;width:16px;height:16px;border:2px solid #f3f3f3;border-top:2px solid #2196f3;border-radius:50%;animation:spin 1s linear infinite}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"] }]
|
|
289
|
+
}], propDecorators: { node: [{
|
|
290
|
+
type: Input,
|
|
291
|
+
args: [{ required: true }]
|
|
292
|
+
}], treeModel: [{
|
|
293
|
+
type: Input,
|
|
294
|
+
args: [{ required: true }]
|
|
295
|
+
}], options: [{
|
|
296
|
+
type: Input,
|
|
297
|
+
args: [{ required: true }]
|
|
298
|
+
}], nodeTemplate: [{
|
|
299
|
+
type: Input
|
|
300
|
+
}] } });
|
|
301
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree-node.component.js","sourceRoot":"","sources":["../../../../../../../projects/concepto-user-controls/src/lib/concepto-tree/components/tree-node/tree-node.component.ts"],"names":[],"mappings":"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","sourcesContent":["// lib/components/tree-node/tree-node.component.ts\r\nimport {\r\n  Component,\r\n  Input,\r\n  TemplateRef,\r\n  ChangeDetectionStrategy,\r\n  computed,\r\n} from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { trigger, state, style, transition, animate } from '@angular/animations';\r\nimport { TreeNode } from '../../core/models/tree-node.model';\r\nimport { TreeModel } from '../../core/models/tree.model';\r\nimport { TreeOptions } from '../../core/models/tree-options.model';\r\nimport { TreeNodeExpanderComponent } from '../tree-node-expander/tree-node-expander.component';\r\nimport { TreeNodeCheckboxComponent } from '../tree-node-checkbox/tree-node-checkbox.component';\r\nimport { TreeNodeContentComponent } from '../tree-node-content/tree-node-content.component';\r\nimport { TreeDragDirective } from '../../directives/tree-drag.directive';\r\nimport { TreeDropDirective } from '../../directives/tree-drop.directive';\r\n\r\n@Component({\r\n  selector: 'tree-node',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    TreeNodeExpanderComponent,\r\n    TreeNodeCheckboxComponent,\r\n    TreeNodeContentComponent,\r\n    TreeDragDirective,\r\n    TreeDropDirective,\r\n  ],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <div \r\n      class=\"tree-node\"\r\n      [class.tree-node-expanded]=\"node.isExpanded()\"\r\n      [class.tree-node-collapsed]=\"node.isCollapsed()\"\r\n      [class.tree-node-active]=\"node.isActive()\"\r\n      [class.tree-node-focused]=\"node.isFocused()\"\r\n      [class.tree-node-leaf]=\"node.isLeaf()\"\r\n      [class.tree-node-loading]=\"node.isLoading()\"\r\n      [style.padding-left.px]=\"paddingLeft()\">\r\n      \r\n      <div class=\"tree-node-wrapper\"\r\n           [treeDrag]=\"node\"\r\n           [treeModel]=\"treeModel\"\r\n           [options]=\"options\">\r\n        \r\n        <!-- Drop zone before node -->\r\n        <div class=\"tree-node-drop-slot tree-node-drop-slot-before\"\r\n             [treeDrop]=\"node\"\r\n             [treeModel]=\"treeModel\"\r\n             [dropPosition]=\"'before'\"\r\n             [options]=\"options\">\r\n        </div>\r\n        \r\n        <!-- Node content wrapper -->\r\n        <div class=\"tree-node-content-wrapper\"\r\n             (click)=\"onNodeClick($event)\"\r\n             (dblclick)=\"onNodeDoubleClick($event)\"\r\n             (contextmenu)=\"onNodeContextMenu($event)\">\r\n          \r\n          <!-- Expander -->\r\n          @if (node.hasChildren) {\r\n            <tree-node-expander\r\n              [node]=\"node\"\r\n              [treeModel]=\"treeModel\"\r\n              (toggle)=\"onToggleExpanded()\">\r\n            </tree-node-expander>\r\n          }\r\n          \r\n          <!-- Checkbox -->\r\n          @if (options.useCheckbox) {\r\n            <tree-node-checkbox\r\n              [node]=\"node\"\r\n              [treeModel]=\"treeModel\"\r\n              [useTriState]=\"options.useTriState ?? false\"\r\n              (change)=\"onCheckboxChange($event)\">\r\n            </tree-node-checkbox>\r\n          }\r\n          \r\n          <!-- Content -->\r\n          <tree-node-content\r\n            [node]=\"node\"\r\n            [treeModel]=\"treeModel\"\r\n            [template]=\"nodeTemplate\"\r\n            [displayField]=\"options.displayField || 'name'\">\r\n          </tree-node-content>\r\n        </div>\r\n        \r\n        <!-- Drop zone after node -->\r\n        <div class=\"tree-node-drop-slot tree-node-drop-slot-after\"\r\n             [treeDrop]=\"node\"\r\n             [treeModel]=\"treeModel\"\r\n             [dropPosition]=\"'after'\"\r\n             [options]=\"options\">\r\n        </div>\r\n      </div>\r\n      \r\n      <!-- Children container -->\r\n      @if (node.isExpanded() && node.children.length > 0) {\r\n        <div class=\"tree-node-children\"\r\n             [@expandCollapse]=\"node.isExpanded() ? 'expanded' : 'collapsed'\">\r\n          @for (child of node.children; track child.id) {\r\n            <tree-node\r\n              [node]=\"child\"\r\n              [treeModel]=\"treeModel\"\r\n              [options]=\"options\"\r\n              [nodeTemplate]=\"nodeTemplate\">\r\n            </tree-node>\r\n          }\r\n        </div>\r\n      }\r\n      \r\n      <!-- Loading indicator -->\r\n      @if (node.isLoading()) {\r\n        <div class=\"tree-node-loading\">\r\n          <span class=\"loading-spinner\"></span>\r\n          Loading...\r\n        </div>\r\n      }\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    .tree-node {\r\n      position: relative;\r\n    }\r\n    \r\n    .tree-node-wrapper {\r\n      position: relative;\r\n      display: flex;\r\n      align-items: center;\r\n      min-height: 22px;\r\n    }\r\n    \r\n    .tree-node-content-wrapper {\r\n      display: flex;\r\n      align-items: center;\r\n      flex: 1;\r\n      cursor: pointer;\r\n      border-radius: 3px;\r\n      padding: 2px 4px;\r\n      transition: background-color 0.15s;\r\n    }\r\n    \r\n    .tree-node-content-wrapper:hover {\r\n      background-color: #f5f5f5;\r\n    }\r\n    \r\n    .tree-node-active .tree-node-content-wrapper {\r\n      background-color: #e3f2fd;\r\n    }\r\n    \r\n    .tree-node-focused .tree-node-content-wrapper {\r\n      outline: 2px solid #2196f3;\r\n      outline-offset: -2px;\r\n    }\r\n    \r\n    .tree-node-children {\r\n      overflow: hidden;\r\n    }\r\n    \r\n    .tree-node-drop-slot {\r\n      position: absolute;\r\n      left: 0;\r\n      right: 0;\r\n      height: 2px;\r\n      z-index: 10;\r\n    }\r\n    \r\n    .tree-node-drop-slot-before {\r\n      top: -1px;\r\n    }\r\n    \r\n    .tree-node-drop-slot-after {\r\n      bottom: -1px;\r\n    }\r\n    \r\n    .tree-node-loading {\r\n      padding: 8px 24px;\r\n      display: flex;\r\n      align-items: center;\r\n      gap: 8px;\r\n      color: #666;\r\n      font-size: 14px;\r\n    }\r\n    \r\n    .loading-spinner {\r\n      display: inline-block;\r\n      width: 16px;\r\n      height: 16px;\r\n      border: 2px solid #f3f3f3;\r\n      border-top: 2px solid #2196f3;\r\n      border-radius: 50%;\r\n      animation: spin 1s linear infinite;\r\n    }\r\n    \r\n    @keyframes spin {\r\n      0% { transform: rotate(0deg); }\r\n      100% { transform: rotate(360deg); }\r\n    }\r\n  `],\r\n  animations: [\r\n    trigger('expandCollapse', [\r\n      state('collapsed', style({\r\n        height: '0',\r\n        overflow: 'hidden',\r\n        opacity: '0'\r\n      })),\r\n      state('expanded', style({\r\n        height: '*',\r\n        overflow: 'visible',\r\n        opacity: '1'\r\n      })),\r\n      transition('collapsed <=> expanded', [\r\n        animate('200ms ease-in-out')\r\n      ])\r\n    ])\r\n  ],\r\n})\r\nexport class TreeNodeComponent {\r\n  @Input({ required: true }) node!: TreeNode;\r\n  @Input({ required: true }) treeModel!: TreeModel;\r\n  @Input({ required: true }) options!: TreeOptions;\r\n  @Input() nodeTemplate?: TemplateRef<any>;\r\n  \r\n  readonly paddingLeft = computed(() => {\r\n    return this.options.levelPadding || 40;\r\n  });\r\n  \r\n  onNodeClick(event: MouseEvent): void {\r\n    const useMetaKey = this.options.useMetaKey !== false;\r\n    const multiSelect = this.options.multiSelect || false;\r\n    \r\n    const isMetaKeyPressed = event.ctrlKey || event.metaKey;\r\n    const shouldMultiSelect = multiSelect && (!useMetaKey || isMetaKeyPressed);\r\n    \r\n    this.treeModel.activateNode(this.node, shouldMultiSelect);\r\n    this.treeModel.focusNode(this.node);\r\n  }\r\n  \r\n  onNodeDoubleClick(event: MouseEvent): void {\r\n    if (this.node.hasChildren) {\r\n      this.node.toggle();\r\n      this.treeModel.expandedNodeIds.update(ids => {\r\n        const newIds = new Set(ids);\r\n        if (this.node.isExpanded()) {\r\n          newIds.add(this.node.id);\r\n        } else {\r\n          newIds.delete(this.node.id);\r\n        }\r\n        return newIds;\r\n      });\r\n    }\r\n  }\r\n  \r\n  onNodeContextMenu(event: MouseEvent): void {\r\n    event.preventDefault();\r\n    // Emit context menu event\r\n  }\r\n  \r\n  onToggleExpanded(): void {\r\n    if (this.node.hasChildren) {\r\n      this.node.toggle();\r\n      \r\n      if (this.node.isExpanded()) {\r\n        // Load children if needed\r\n        if (this.options.getChildren && this.node.children.length === 0) {\r\n          this.treeModel.loadChildren(this.node);\r\n        }\r\n        this.treeModel.expandNode(this.node);\r\n      } else {\r\n        this.treeModel.collapseNode(this.node);\r\n      }\r\n    }\r\n  }\r\n  \r\n  onCheckboxChange(checked: boolean): void {\r\n    this.treeModel.selectNode(this.node, this.options.multiSelect || false);\r\n  }\r\n}"]}
|
package/esm2022/lib/concepto-tree/components/tree-node-checkbox/tree-node-checkbox.component.mjs
ADDED
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
// lib/components/tree-node-checkbox/tree-node-checkbox.component.ts
|
|
2
|
+
import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy, computed, } from '@angular/core';
|
|
3
|
+
import { CommonModule } from '@angular/common';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
export class TreeNodeCheckboxComponent {
|
|
6
|
+
node;
|
|
7
|
+
treeModel;
|
|
8
|
+
useTriState = false;
|
|
9
|
+
change = new EventEmitter();
|
|
10
|
+
isChecked = computed(() => {
|
|
11
|
+
return this.treeModel.selectedNodeIds().has(this.node.id);
|
|
12
|
+
});
|
|
13
|
+
isIndeterminate = computed(() => {
|
|
14
|
+
if (!this.useTriState || !this.node.hasChildren) {
|
|
15
|
+
return false;
|
|
16
|
+
}
|
|
17
|
+
const selectedIds = this.treeModel.selectedNodeIds();
|
|
18
|
+
const descendants = this.node.getDescendants();
|
|
19
|
+
if (descendants.length === 0)
|
|
20
|
+
return false;
|
|
21
|
+
const selectedCount = descendants.filter(d => selectedIds.has(d.id)).length;
|
|
22
|
+
return selectedCount > 0 && selectedCount < descendants.length;
|
|
23
|
+
});
|
|
24
|
+
onCheckboxClick(event) {
|
|
25
|
+
event.stopPropagation();
|
|
26
|
+
}
|
|
27
|
+
onCheckboxChange(event) {
|
|
28
|
+
event.stopPropagation();
|
|
29
|
+
const checked = event.target.checked;
|
|
30
|
+
if (this.useTriState) {
|
|
31
|
+
// Update node and all descendants
|
|
32
|
+
this.updateNodeAndDescendants(this.node, checked);
|
|
33
|
+
}
|
|
34
|
+
else {
|
|
35
|
+
this.change.emit(checked);
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
updateNodeAndDescendants(node, checked) {
|
|
39
|
+
this.treeModel.selectedNodeIds.update(ids => {
|
|
40
|
+
const newIds = new Set(ids);
|
|
41
|
+
if (checked) {
|
|
42
|
+
newIds.add(node.id);
|
|
43
|
+
node.getDescendants().forEach(d => newIds.add(d.id));
|
|
44
|
+
}
|
|
45
|
+
else {
|
|
46
|
+
newIds.delete(node.id);
|
|
47
|
+
node.getDescendants().forEach(d => newIds.delete(d.id));
|
|
48
|
+
}
|
|
49
|
+
return newIds;
|
|
50
|
+
});
|
|
51
|
+
this.change.emit(checked);
|
|
52
|
+
}
|
|
53
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeNodeCheckboxComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
54
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: TreeNodeCheckboxComponent, isStandalone: true, selector: "tree-node-checkbox", inputs: { node: "node", treeModel: "treeModel", useTriState: "useTriState" }, outputs: { change: "change" }, ngImport: i0, template: `
|
|
55
|
+
<div class="tree-node-checkbox"
|
|
56
|
+
(click)="onCheckboxClick($event)">
|
|
57
|
+
<input
|
|
58
|
+
type="checkbox"
|
|
59
|
+
[checked]="isChecked()"
|
|
60
|
+
[indeterminate]="isIndeterminate()"
|
|
61
|
+
(change)="onCheckboxChange($event)"
|
|
62
|
+
(click)="$event.stopPropagation()">
|
|
63
|
+
</div>
|
|
64
|
+
`, isInline: true, styles: [".tree-node-checkbox{display:flex;align-items:center;justify-content:center;width:20px;height:20px;flex-shrink:0}input[type=checkbox]{cursor:pointer;width:16px;height:16px;margin:0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
65
|
+
}
|
|
66
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeNodeCheckboxComponent, decorators: [{
|
|
67
|
+
type: Component,
|
|
68
|
+
args: [{ selector: 'tree-node-checkbox', standalone: true, imports: [CommonModule], changeDetection: ChangeDetectionStrategy.OnPush, template: `
|
|
69
|
+
<div class="tree-node-checkbox"
|
|
70
|
+
(click)="onCheckboxClick($event)">
|
|
71
|
+
<input
|
|
72
|
+
type="checkbox"
|
|
73
|
+
[checked]="isChecked()"
|
|
74
|
+
[indeterminate]="isIndeterminate()"
|
|
75
|
+
(change)="onCheckboxChange($event)"
|
|
76
|
+
(click)="$event.stopPropagation()">
|
|
77
|
+
</div>
|
|
78
|
+
`, styles: [".tree-node-checkbox{display:flex;align-items:center;justify-content:center;width:20px;height:20px;flex-shrink:0}input[type=checkbox]{cursor:pointer;width:16px;height:16px;margin:0}\n"] }]
|
|
79
|
+
}], propDecorators: { node: [{
|
|
80
|
+
type: Input,
|
|
81
|
+
args: [{ required: true }]
|
|
82
|
+
}], treeModel: [{
|
|
83
|
+
type: Input,
|
|
84
|
+
args: [{ required: true }]
|
|
85
|
+
}], useTriState: [{
|
|
86
|
+
type: Input
|
|
87
|
+
}], change: [{
|
|
88
|
+
type: Output
|
|
89
|
+
}] } });
|
|
90
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJlZS1ub2RlLWNoZWNrYm94LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbmNlcHRvLXVzZXItY29udHJvbHMvc3JjL2xpYi9jb25jZXB0by10cmVlL2NvbXBvbmVudHMvdHJlZS1ub2RlLWNoZWNrYm94L3RyZWUtbm9kZS1jaGVja2JveC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsb0VBQW9FO0FBQ3BFLE9BQU8sRUFDTCxTQUFTLEVBQ1QsS0FBSyxFQUNMLE1BQU0sRUFDTixZQUFZLEVBQ1osdUJBQXVCLEVBQ3ZCLFFBQVEsR0FDVCxNQUFNLGVBQWUsQ0FBQztBQUN2QixPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7O0FBc0MvQyxNQUFNLE9BQU8seUJBQXlCO0lBQ1QsSUFBSSxDQUFZO0lBQ2hCLFNBQVMsQ0FBYTtJQUN4QyxXQUFXLEdBQUcsS0FBSyxDQUFDO0lBQ25CLE1BQU0sR0FBRyxJQUFJLFlBQVksRUFBVyxDQUFDO0lBRXRDLFNBQVMsR0FBRyxRQUFRLENBQUMsR0FBRyxFQUFFO1FBQ2pDLE9BQU8sSUFBSSxDQUFDLFNBQVMsQ0FBQyxlQUFlLEVBQUUsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQztJQUM1RCxDQUFDLENBQUMsQ0FBQztJQUVNLGVBQWUsR0FBRyxRQUFRLENBQUMsR0FBRyxFQUFFO1FBQ3ZDLElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQztZQUNoRCxPQUFPLEtBQUssQ0FBQztRQUNmLENBQUM7UUFFRCxNQUFNLFdBQVcsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLGVBQWUsRUFBRSxDQUFDO1FBQ3JELE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsY0FBYyxFQUFFLENBQUM7UUFFL0MsSUFBSSxXQUFXLENBQUMsTUFBTSxLQUFLLENBQUM7WUFBRSxPQUFPLEtBQUssQ0FBQztRQUUzQyxNQUFNLGFBQWEsR0FBRyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQzNDLFdBQVcsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUN0QixDQUFDLE1BQU0sQ0FBQztRQUVULE9BQU8sYUFBYSxHQUFHLENBQUMsSUFBSSxhQUFhLEdBQUcsV0FBVyxDQUFDLE1BQU0sQ0FBQztJQUNqRSxDQUFDLENBQUMsQ0FBQztJQUVILGVBQWUsQ0FBQyxLQUFpQjtRQUMvQixLQUFLLENBQUMsZUFBZSxFQUFFLENBQUM7SUFDMUIsQ0FBQztJQUVELGdCQUFnQixDQUFDLEtBQVk7UUFDM0IsS0FBSyxDQUFDLGVBQWUsRUFBRSxDQUFDO1FBQ3hCLE1BQU0sT0FBTyxHQUFJLEtBQUssQ0FBQyxNQUEyQixDQUFDLE9BQU8sQ0FBQztRQUUzRCxJQUFJLElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQztZQUNyQixrQ0FBa0M7WUFDbEMsSUFBSSxDQUFDLHdCQUF3QixDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDLENBQUM7UUFDcEQsQ0FBQzthQUFNLENBQUM7WUFDTixJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUM1QixDQUFDO0lBQ0gsQ0FBQztJQUVPLHdCQUF3QixDQUFDLElBQWMsRUFBRSxPQUFnQjtRQUMvRCxJQUFJLENBQUMsU0FBUyxDQUFDLGVBQWUsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLEVBQUU7WUFDMUMsTUFBTSxNQUFNLEdBQUcsSUFBSSxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUM7WUFFNUIsSUFBSSxPQUFPLEVBQUUsQ0FBQztnQkFDWixNQUFNLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQztnQkFDcEIsSUFBSSxDQUFDLGNBQWMsRUFBRSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7WUFDdkQsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLE1BQU0sQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDO2dCQUN2QixJQUFJLENBQUMsY0FBYyxFQUFFLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQztZQUMxRCxDQUFDO1lBRUQsT0FBTyxNQUFNLENBQUM7UUFDaEIsQ0FBQyxDQUFDLENBQUM7UUFFSCxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUM1QixDQUFDO3dHQTNEVSx5QkFBeUI7NEZBQXpCLHlCQUF5QiwyTEE3QjFCOzs7Ozs7Ozs7O0dBVVQsK1BBWlMsWUFBWTs7NEZBK0JYLHlCQUF5QjtrQkFsQ3JDLFNBQVM7K0JBQ0Usb0JBQW9CLGNBQ2xCLElBQUksV0FDUCxDQUFDLFlBQVksQ0FBQyxtQkFDTix1QkFBdUIsQ0FBQyxNQUFNLFlBQ3JDOzs7Ozs7Ozs7O0dBVVQ7OEJBb0IwQixJQUFJO3NCQUE5QixLQUFLO3VCQUFDLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRTtnQkFDRSxTQUFTO3NCQUFuQyxLQUFLO3VCQUFDLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRTtnQkFDaEIsV0FBVztzQkFBbkIsS0FBSztnQkFDSSxNQUFNO3NCQUFmLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyIvLyBsaWIvY29tcG9uZW50cy90cmVlLW5vZGUtY2hlY2tib3gvdHJlZS1ub2RlLWNoZWNrYm94LmNvbXBvbmVudC50c1xyXG5pbXBvcnQge1xyXG4gIENvbXBvbmVudCxcclxuICBJbnB1dCxcclxuICBPdXRwdXQsXHJcbiAgRXZlbnRFbWl0dGVyLFxyXG4gIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LFxyXG4gIGNvbXB1dGVkLFxyXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xyXG5pbXBvcnQgeyBUcmVlTm9kZSB9IGZyb20gJy4uLy4uL2NvcmUvbW9kZWxzL3RyZWUtbm9kZS5tb2RlbCc7XHJcbmltcG9ydCB7IFRyZWVNb2RlbCB9IGZyb20gJy4uLy4uL2NvcmUvbW9kZWxzL3RyZWUubW9kZWwnO1xyXG5cclxuQENvbXBvbmVudCh7XHJcbiAgc2VsZWN0b3I6ICd0cmVlLW5vZGUtY2hlY2tib3gnLFxyXG4gIHN0YW5kYWxvbmU6IHRydWUsXHJcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZV0sXHJcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXHJcbiAgdGVtcGxhdGU6IGBcclxuICAgIDxkaXYgY2xhc3M9XCJ0cmVlLW5vZGUtY2hlY2tib3hcIlxyXG4gICAgICAgICAoY2xpY2spPVwib25DaGVja2JveENsaWNrKCRldmVudClcIj5cclxuICAgICAgPGlucHV0IFxyXG4gICAgICAgIHR5cGU9XCJjaGVja2JveFwiXHJcbiAgICAgICAgW2NoZWNrZWRdPVwiaXNDaGVja2VkKClcIlxyXG4gICAgICAgIFtpbmRldGVybWluYXRlXT1cImlzSW5kZXRlcm1pbmF0ZSgpXCJcclxuICAgICAgICAoY2hhbmdlKT1cIm9uQ2hlY2tib3hDaGFuZ2UoJGV2ZW50KVwiXHJcbiAgICAgICAgKGNsaWNrKT1cIiRldmVudC5zdG9wUHJvcGFnYXRpb24oKVwiPlxyXG4gICAgPC9kaXY+XHJcbiAgYCxcclxuICBzdHlsZXM6IFtgXHJcbiAgICAudHJlZS1ub2RlLWNoZWNrYm94IHtcclxuICAgICAgZGlzcGxheTogZmxleDtcclxuICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcclxuICAgICAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XHJcbiAgICAgIHdpZHRoOiAyMHB4O1xyXG4gICAgICBoZWlnaHQ6IDIwcHg7XHJcbiAgICAgIGZsZXgtc2hyaW5rOiAwO1xyXG4gICAgfVxyXG4gICAgXHJcbiAgICBpbnB1dFt0eXBlPVwiY2hlY2tib3hcIl0ge1xyXG4gICAgICBjdXJzb3I6IHBvaW50ZXI7XHJcbiAgICAgIHdpZHRoOiAxNnB4O1xyXG4gICAgICBoZWlnaHQ6IDE2cHg7XHJcbiAgICAgIG1hcmdpbjogMDtcclxuICAgIH1cclxuICBgXSxcclxufSlcclxuZXhwb3J0IGNsYXNzIFRyZWVOb2RlQ2hlY2tib3hDb21wb25lbnQge1xyXG4gIEBJbnB1dCh7IHJlcXVpcmVkOiB0cnVlIH0pIG5vZGUhOiBUcmVlTm9kZTtcclxuICBASW5wdXQoeyByZXF1aXJlZDogdHJ1ZSB9KSB0cmVlTW9kZWwhOiBUcmVlTW9kZWw7XHJcbiAgQElucHV0KCkgdXNlVHJpU3RhdGUgPSBmYWxzZTtcclxuICBAT3V0cHV0KCkgY2hhbmdlID0gbmV3IEV2ZW50RW1pdHRlcjxib29sZWFuPigpO1xyXG4gIFxyXG4gIHJlYWRvbmx5IGlzQ2hlY2tlZCA9IGNvbXB1dGVkKCgpID0+IHtcclxuICAgIHJldHVybiB0aGlzLnRyZWVNb2RlbC5zZWxlY3RlZE5vZGVJZHMoKS5oYXModGhpcy5ub2RlLmlkKTtcclxuICB9KTtcclxuICBcclxuICByZWFkb25seSBpc0luZGV0ZXJtaW5hdGUgPSBjb21wdXRlZCgoKSA9PiB7XHJcbiAgICBpZiAoIXRoaXMudXNlVHJpU3RhdGUgfHwgIXRoaXMubm9kZS5oYXNDaGlsZHJlbikge1xyXG4gICAgICByZXR1cm4gZmFsc2U7XHJcbiAgICB9XHJcbiAgICBcclxuICAgIGNvbnN0IHNlbGVjdGVkSWRzID0gdGhpcy50cmVlTW9kZWwuc2VsZWN0ZWROb2RlSWRzKCk7XHJcbiAgICBjb25zdCBkZXNjZW5kYW50cyA9IHRoaXMubm9kZS5nZXREZXNjZW5kYW50cygpO1xyXG4gICAgXHJcbiAgICBpZiAoZGVzY2VuZGFudHMubGVuZ3RoID09PSAwKSByZXR1cm4gZmFsc2U7XHJcbiAgICBcclxuICAgIGNvbnN0IHNlbGVjdGVkQ291bnQgPSBkZXNjZW5kYW50cy5maWx0ZXIoZCA9PiBcclxuICAgICAgc2VsZWN0ZWRJZHMuaGFzKGQuaWQpXHJcbiAgICApLmxlbmd0aDtcclxuICAgIFxyXG4gICAgcmV0dXJuIHNlbGVjdGVkQ291bnQgPiAwICYmIHNlbGVjdGVkQ291bnQgPCBkZXNjZW5kYW50cy5sZW5ndGg7XHJcbiAgfSk7XHJcbiAgXHJcbiAgb25DaGVja2JveENsaWNrKGV2ZW50OiBNb3VzZUV2ZW50KTogdm9pZCB7XHJcbiAgICBldmVudC5zdG9wUHJvcGFnYXRpb24oKTtcclxuICB9XHJcbiAgXHJcbiAgb25DaGVja2JveENoYW5nZShldmVudDogRXZlbnQpOiB2b2lkIHtcclxuICAgIGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpO1xyXG4gICAgY29uc3QgY2hlY2tlZCA9IChldmVudC50YXJnZXQgYXMgSFRNTElucHV0RWxlbWVudCkuY2hlY2tlZDtcclxuICAgIFxyXG4gICAgaWYgKHRoaXMudXNlVHJpU3RhdGUpIHtcclxuICAgICAgLy8gVXBkYXRlIG5vZGUgYW5kIGFsbCBkZXNjZW5kYW50c1xyXG4gICAgICB0aGlzLnVwZGF0ZU5vZGVBbmREZXNjZW5kYW50cyh0aGlzLm5vZGUsIGNoZWNrZWQpO1xyXG4gICAgfSBlbHNlIHtcclxuICAgICAgdGhpcy5jaGFuZ2UuZW1pdChjaGVja2VkKTtcclxuICAgIH1cclxuICB9XHJcbiAgXHJcbiAgcHJpdmF0ZSB1cGRhdGVOb2RlQW5kRGVzY2VuZGFudHMobm9kZTogVHJlZU5vZGUsIGNoZWNrZWQ6IGJvb2xlYW4pOiB2b2lkIHtcclxuICAgIHRoaXMudHJlZU1vZGVsLnNlbGVjdGVkTm9kZUlkcy51cGRhdGUoaWRzID0+IHtcclxuICAgICAgY29uc3QgbmV3SWRzID0gbmV3IFNldChpZHMpO1xyXG4gICAgICBcclxuICAgICAgaWYgKGNoZWNrZWQpIHtcclxuICAgICAgICBuZXdJZHMuYWRkKG5vZGUuaWQpO1xyXG4gICAgICAgIG5vZGUuZ2V0RGVzY2VuZGFudHMoKS5mb3JFYWNoKGQgPT4gbmV3SWRzLmFkZChkLmlkKSk7XHJcbiAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgbmV3SWRzLmRlbGV0ZShub2RlLmlkKTtcclxuICAgICAgICBub2RlLmdldERlc2NlbmRhbnRzKCkuZm9yRWFjaChkID0+IG5ld0lkcy5kZWxldGUoZC5pZCkpO1xyXG4gICAgICB9XHJcbiAgICAgIFxyXG4gICAgICByZXR1cm4gbmV3SWRzO1xyXG4gICAgfSk7XHJcbiAgICBcclxuICAgIHRoaXMuY2hhbmdlLmVtaXQoY2hlY2tlZCk7XHJcbiAgfVxyXG59Il19
|
package/esm2022/lib/concepto-tree/components/tree-node-content/tree-node-content.component.mjs
ADDED
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
// lib/components/tree-node-content/tree-node-content.component.ts
|
|
2
|
+
import { Component, Input, ChangeDetectionStrategy, } from '@angular/core';
|
|
3
|
+
import { CommonModule } from '@angular/common';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/common";
|
|
6
|
+
export class TreeNodeContentComponent {
|
|
7
|
+
node;
|
|
8
|
+
treeModel;
|
|
9
|
+
template;
|
|
10
|
+
displayField = 'name';
|
|
11
|
+
NgOnInit() {
|
|
12
|
+
console.log("Tree node:", this.node);
|
|
13
|
+
}
|
|
14
|
+
get templateContext() {
|
|
15
|
+
console.log("Tree node:", this.node);
|
|
16
|
+
return {
|
|
17
|
+
$implicit: this.node,
|
|
18
|
+
node: this.node,
|
|
19
|
+
treeModel: this.treeModel,
|
|
20
|
+
};
|
|
21
|
+
}
|
|
22
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeNodeContentComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
23
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: TreeNodeContentComponent, isStandalone: true, selector: "tree-node-content", inputs: { node: "node", treeModel: "treeModel", template: "template", displayField: "displayField" }, ngImport: i0, template: `
|
|
24
|
+
@if (template) {
|
|
25
|
+
<ng-container
|
|
26
|
+
*ngTemplateOutlet="template; context: templateContext">
|
|
27
|
+
</ng-container>
|
|
28
|
+
} @else {
|
|
29
|
+
<span class="tree-node-content-default">
|
|
30
|
+
@if (node.icon) {
|
|
31
|
+
<img [src]="node.icon" alt="" class="tree-node-icon" />
|
|
32
|
+
}
|
|
33
|
+
<span class="tree-node-text">{{ node.data[displayField] }}</span>
|
|
34
|
+
</span>
|
|
35
|
+
}
|
|
36
|
+
`, isInline: true, styles: [".tree-node-content-default{flex:1;padding:0 4px;display:flex;align-items:center;gap:6px;overflow:hidden;-webkit-user-select:none;user-select:none}.tree-node-icon{width:16px;height:16px;flex-shrink:0;object-fit:contain}.tree-node-text{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
37
|
+
}
|
|
38
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeNodeContentComponent, decorators: [{
|
|
39
|
+
type: Component,
|
|
40
|
+
args: [{ selector: 'tree-node-content', standalone: true, imports: [CommonModule], changeDetection: ChangeDetectionStrategy.OnPush, template: `
|
|
41
|
+
@if (template) {
|
|
42
|
+
<ng-container
|
|
43
|
+
*ngTemplateOutlet="template; context: templateContext">
|
|
44
|
+
</ng-container>
|
|
45
|
+
} @else {
|
|
46
|
+
<span class="tree-node-content-default">
|
|
47
|
+
@if (node.icon) {
|
|
48
|
+
<img [src]="node.icon" alt="" class="tree-node-icon" />
|
|
49
|
+
}
|
|
50
|
+
<span class="tree-node-text">{{ node.data[displayField] }}</span>
|
|
51
|
+
</span>
|
|
52
|
+
}
|
|
53
|
+
`, styles: [".tree-node-content-default{flex:1;padding:0 4px;display:flex;align-items:center;gap:6px;overflow:hidden;-webkit-user-select:none;user-select:none}.tree-node-icon{width:16px;height:16px;flex-shrink:0;object-fit:contain}.tree-node-text{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}\n"] }]
|
|
54
|
+
}], propDecorators: { node: [{
|
|
55
|
+
type: Input,
|
|
56
|
+
args: [{ required: true }]
|
|
57
|
+
}], treeModel: [{
|
|
58
|
+
type: Input,
|
|
59
|
+
args: [{ required: true }]
|
|
60
|
+
}], template: [{
|
|
61
|
+
type: Input
|
|
62
|
+
}], displayField: [{
|
|
63
|
+
type: Input
|
|
64
|
+
}] } });
|
|
65
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJlZS1ub2RlLWNvbnRlbnQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY29uY2VwdG8tdXNlci1jb250cm9scy9zcmMvbGliL2NvbmNlcHRvLXRyZWUvY29tcG9uZW50cy90cmVlLW5vZGUtY29udGVudC90cmVlLW5vZGUtY29udGVudC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsa0VBQWtFO0FBQ2xFLE9BQU8sRUFDTCxTQUFTLEVBQ1QsS0FBSyxFQUVMLHVCQUF1QixHQUN4QixNQUFNLGVBQWUsQ0FBQztBQUN2QixPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7OztBQWdEL0MsTUFBTSxPQUFPLHdCQUF3QjtJQUNSLElBQUksQ0FBWTtJQUNoQixTQUFTLENBQWE7SUFDeEMsUUFBUSxDQUFvQjtJQUM1QixZQUFZLEdBQUcsTUFBTSxDQUFDO0lBRS9CLFFBQVE7UUFDTixPQUFPLENBQUMsR0FBRyxDQUFDLFlBQVksRUFBRSxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDdkMsQ0FBQztJQUVELElBQUksZUFBZTtRQUNqQixPQUFPLENBQUMsR0FBRyxDQUFDLFlBQVksRUFBRSxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDckMsT0FBTztZQUNMLFNBQVMsRUFBRSxJQUFJLENBQUMsSUFBSTtZQUNwQixJQUFJLEVBQUUsSUFBSSxDQUFDLElBQUk7WUFDZixTQUFTLEVBQUUsSUFBSSxDQUFDLFNBQVM7U0FDMUIsQ0FBQztJQUNKLENBQUM7d0dBakJVLHdCQUF3Qjs0RkFBeEIsd0JBQXdCLG1MQXZDekI7Ozs7Ozs7Ozs7Ozs7R0FhVCwrV0FmUyxZQUFZOzs0RkF5Q1gsd0JBQXdCO2tCQTVDcEMsU0FBUzsrQkFDRSxtQkFBbUIsY0FDakIsSUFBSSxXQUNQLENBQUMsWUFBWSxDQUFDLG1CQUNOLHVCQUF1QixDQUFDLE1BQU0sWUFDckM7Ozs7Ozs7Ozs7Ozs7R0FhVDs4QkEyQjBCLElBQUk7c0JBQTlCLEtBQUs7dUJBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFO2dCQUNFLFNBQVM7c0JBQW5DLEtBQUs7dUJBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFO2dCQUNoQixRQUFRO3NCQUFoQixLQUFLO2dCQUNHLFlBQVk7c0JBQXBCLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBsaWIvY29tcG9uZW50cy90cmVlLW5vZGUtY29udGVudC90cmVlLW5vZGUtY29udGVudC5jb21wb25lbnQudHNcclxuaW1wb3J0IHtcclxuICBDb21wb25lbnQsXHJcbiAgSW5wdXQsXHJcbiAgVGVtcGxhdGVSZWYsXHJcbiAgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksXHJcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XHJcbmltcG9ydCB7IFRyZWVOb2RlIH0gZnJvbSAnLi4vLi4vY29yZS9tb2RlbHMvdHJlZS1ub2RlLm1vZGVsJztcclxuaW1wb3J0IHsgVHJlZU1vZGVsIH0gZnJvbSAnLi4vLi4vY29yZS9tb2RlbHMvdHJlZS5tb2RlbCc7XHJcblxyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogJ3RyZWUtbm9kZS1jb250ZW50JyxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGVdLFxyXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxyXG4gIHRlbXBsYXRlOiBgXHJcbiAgICBAaWYgKHRlbXBsYXRlKSB7XHJcbiAgICAgIDxuZy1jb250YWluZXJcclxuICAgICAgICAqbmdUZW1wbGF0ZU91dGxldD1cInRlbXBsYXRlOyBjb250ZXh0OiB0ZW1wbGF0ZUNvbnRleHRcIj5cclxuICAgICAgPC9uZy1jb250YWluZXI+XHJcbiAgICB9IEBlbHNlIHtcclxuICAgICAgPHNwYW4gY2xhc3M9XCJ0cmVlLW5vZGUtY29udGVudC1kZWZhdWx0XCI+XHJcbiAgICAgICAgQGlmIChub2RlLmljb24pIHtcclxuICAgICAgICAgIDxpbWcgW3NyY109XCJub2RlLmljb25cIiBhbHQ9XCJcIiBjbGFzcz1cInRyZWUtbm9kZS1pY29uXCIgLz5cclxuICAgICAgICB9XHJcbiAgICAgICAgPHNwYW4gY2xhc3M9XCJ0cmVlLW5vZGUtdGV4dFwiPnt7IG5vZGUuZGF0YVtkaXNwbGF5RmllbGRdIH19PC9zcGFuPlxyXG4gICAgICA8L3NwYW4+XHJcbiAgICB9XHJcbiAgYCxcclxuICBzdHlsZXM6IFtgXHJcbiAgICAudHJlZS1ub2RlLWNvbnRlbnQtZGVmYXVsdCB7XHJcbiAgICAgIGZsZXg6IDE7XHJcbiAgICAgIHBhZGRpbmc6IDAgNHB4O1xyXG4gICAgICBkaXNwbGF5OiBmbGV4O1xyXG4gICAgICBhbGlnbi1pdGVtczogY2VudGVyO1xyXG4gICAgICBnYXA6IDZweDtcclxuICAgICAgb3ZlcmZsb3c6IGhpZGRlbjtcclxuICAgICAgdXNlci1zZWxlY3Q6IG5vbmU7XHJcbiAgICB9XHJcblxyXG4gICAgLnRyZWUtbm9kZS1pY29uIHtcclxuICAgICAgd2lkdGg6IDE2cHg7XHJcbiAgICAgIGhlaWdodDogMTZweDtcclxuICAgICAgZmxleC1zaHJpbms6IDA7XHJcbiAgICAgIG9iamVjdC1maXQ6IGNvbnRhaW47XHJcbiAgICB9XHJcblxyXG4gICAgLnRyZWUtbm9kZS10ZXh0IHtcclxuICAgICAgb3ZlcmZsb3c6IGhpZGRlbjtcclxuICAgICAgdGV4dC1vdmVyZmxvdzogZWxsaXBzaXM7XHJcbiAgICAgIHdoaXRlLXNwYWNlOiBub3dyYXA7XHJcbiAgICB9XHJcbiAgYF0sXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBUcmVlTm9kZUNvbnRlbnRDb21wb25lbnQge1xyXG4gIEBJbnB1dCh7IHJlcXVpcmVkOiB0cnVlIH0pIG5vZGUhOiBUcmVlTm9kZTtcclxuICBASW5wdXQoeyByZXF1aXJlZDogdHJ1ZSB9KSB0cmVlTW9kZWwhOiBUcmVlTW9kZWw7XHJcbiAgQElucHV0KCkgdGVtcGxhdGU/OiBUZW1wbGF0ZVJlZjxhbnk+O1xyXG4gIEBJbnB1dCgpIGRpc3BsYXlGaWVsZCA9ICduYW1lJztcclxuICBcclxuICBOZ09uSW5pdCgpe1xyXG4gICAgY29uc29sZS5sb2coXCJUcmVlIG5vZGU6XCIsIHRoaXMubm9kZSk7XHJcbiAgfVxyXG5cclxuICBnZXQgdGVtcGxhdGVDb250ZXh0KCkge1xyXG4gICAgY29uc29sZS5sb2coXCJUcmVlIG5vZGU6XCIsIHRoaXMubm9kZSk7XHJcbiAgICByZXR1cm4ge1xyXG4gICAgICAkaW1wbGljaXQ6IHRoaXMubm9kZSxcclxuICAgICAgbm9kZTogdGhpcy5ub2RlLFxyXG4gICAgICB0cmVlTW9kZWw6IHRoaXMudHJlZU1vZGVsLFxyXG4gICAgfTtcclxuICB9XHJcbn0iXX0=
|
package/esm2022/lib/concepto-tree/components/tree-node-expander/tree-node-expander.component.mjs
ADDED
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
// lib/components/tree-node-expander/tree-node-expander.component.ts
|
|
2
|
+
import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy, } from '@angular/core';
|
|
3
|
+
import { CommonModule } from '@angular/common';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
export class TreeNodeExpanderComponent {
|
|
6
|
+
node;
|
|
7
|
+
treeModel;
|
|
8
|
+
toggle = new EventEmitter();
|
|
9
|
+
onExpanderClick(event) {
|
|
10
|
+
event.stopPropagation();
|
|
11
|
+
this.toggle.emit();
|
|
12
|
+
}
|
|
13
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeNodeExpanderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
14
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: TreeNodeExpanderComponent, isStandalone: true, selector: "tree-node-expander", inputs: { node: "node", treeModel: "treeModel" }, outputs: { toggle: "toggle" }, ngImport: i0, template: `
|
|
15
|
+
<div class="tree-node-expander"
|
|
16
|
+
[class.tree-node-expander-loading]="node.isLoading()"
|
|
17
|
+
(click)="onExpanderClick($event)">
|
|
18
|
+
@if (node.isLoading()) {
|
|
19
|
+
<span class="expander-spinner"></span>
|
|
20
|
+
} @else if (node.hasChildren) {
|
|
21
|
+
<svg
|
|
22
|
+
class="expander-icon"
|
|
23
|
+
[class.expander-icon-expanded]="node.isExpanded()"
|
|
24
|
+
width="16"
|
|
25
|
+
height="16"
|
|
26
|
+
viewBox="0 0 16 16">
|
|
27
|
+
<path d="M6 4l4 4-4 4"
|
|
28
|
+
fill="none"
|
|
29
|
+
stroke="currentColor"
|
|
30
|
+
stroke-width="2"
|
|
31
|
+
stroke-linecap="round"/>
|
|
32
|
+
</svg>
|
|
33
|
+
} @else {
|
|
34
|
+
<span class="expander-placeholder"></span>
|
|
35
|
+
}
|
|
36
|
+
</div>
|
|
37
|
+
`, isInline: true, styles: [".tree-node-expander{display:flex;align-items:center;justify-content:center;width:20px;height:20px;cursor:pointer;border-radius:3px;transition:background-color .15s;flex-shrink:0}.tree-node-expander:hover{background-color:#0000000d}.expander-icon{transition:transform .2s ease;transform:rotate(0)}.expander-icon-expanded{transform:rotate(90deg)}.expander-placeholder{width:16px;height:16px}.expander-spinner{display:inline-block;width:14px;height:14px;border:2px solid #f3f3f3;border-top:2px solid #2196f3;border-radius:50%;animation:spin 1s linear infinite}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
38
|
+
}
|
|
39
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeNodeExpanderComponent, decorators: [{
|
|
40
|
+
type: Component,
|
|
41
|
+
args: [{ selector: 'tree-node-expander', standalone: true, imports: [CommonModule], changeDetection: ChangeDetectionStrategy.OnPush, template: `
|
|
42
|
+
<div class="tree-node-expander"
|
|
43
|
+
[class.tree-node-expander-loading]="node.isLoading()"
|
|
44
|
+
(click)="onExpanderClick($event)">
|
|
45
|
+
@if (node.isLoading()) {
|
|
46
|
+
<span class="expander-spinner"></span>
|
|
47
|
+
} @else if (node.hasChildren) {
|
|
48
|
+
<svg
|
|
49
|
+
class="expander-icon"
|
|
50
|
+
[class.expander-icon-expanded]="node.isExpanded()"
|
|
51
|
+
width="16"
|
|
52
|
+
height="16"
|
|
53
|
+
viewBox="0 0 16 16">
|
|
54
|
+
<path d="M6 4l4 4-4 4"
|
|
55
|
+
fill="none"
|
|
56
|
+
stroke="currentColor"
|
|
57
|
+
stroke-width="2"
|
|
58
|
+
stroke-linecap="round"/>
|
|
59
|
+
</svg>
|
|
60
|
+
} @else {
|
|
61
|
+
<span class="expander-placeholder"></span>
|
|
62
|
+
}
|
|
63
|
+
</div>
|
|
64
|
+
`, styles: [".tree-node-expander{display:flex;align-items:center;justify-content:center;width:20px;height:20px;cursor:pointer;border-radius:3px;transition:background-color .15s;flex-shrink:0}.tree-node-expander:hover{background-color:#0000000d}.expander-icon{transition:transform .2s ease;transform:rotate(0)}.expander-icon-expanded{transform:rotate(90deg)}.expander-placeholder{width:16px;height:16px}.expander-spinner{display:inline-block;width:14px;height:14px;border:2px solid #f3f3f3;border-top:2px solid #2196f3;border-radius:50%;animation:spin 1s linear infinite}@keyframes spin{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"] }]
|
|
65
|
+
}], propDecorators: { node: [{
|
|
66
|
+
type: Input,
|
|
67
|
+
args: [{ required: true }]
|
|
68
|
+
}], treeModel: [{
|
|
69
|
+
type: Input,
|
|
70
|
+
args: [{ required: true }]
|
|
71
|
+
}], toggle: [{
|
|
72
|
+
type: Output
|
|
73
|
+
}] } });
|
|
74
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJlZS1ub2RlLWV4cGFuZGVyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbmNlcHRvLXVzZXItY29udHJvbHMvc3JjL2xpYi9jb25jZXB0by10cmVlL2NvbXBvbmVudHMvdHJlZS1ub2RlLWV4cGFuZGVyL3RyZWUtbm9kZS1leHBhbmRlci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsb0VBQW9FO0FBQ3BFLE9BQU8sRUFDTCxTQUFTLEVBQ1QsS0FBSyxFQUNMLE1BQU0sRUFDTixZQUFZLEVBQ1osdUJBQXVCLEdBQ3hCLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQzs7QUFnRi9DLE1BQU0sT0FBTyx5QkFBeUI7SUFDVCxJQUFJLENBQVk7SUFDaEIsU0FBUyxDQUFhO0lBQ3ZDLE1BQU0sR0FBRyxJQUFJLFlBQVksRUFBUSxDQUFDO0lBRTVDLGVBQWUsQ0FBQyxLQUFpQjtRQUMvQixLQUFLLENBQUMsZUFBZSxFQUFFLENBQUM7UUFDeEIsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUNyQixDQUFDO3dHQVJVLHlCQUF5Qjs0RkFBekIseUJBQXlCLCtKQXZFMUI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBdUJULDRyQkF6QlMsWUFBWTs7NEZBeUVYLHlCQUF5QjtrQkE1RXJDLFNBQVM7K0JBQ0Usb0JBQW9CLGNBQ2xCLElBQUksV0FDUCxDQUFDLFlBQVksQ0FBQyxtQkFDTix1QkFBdUIsQ0FBQyxNQUFNLFlBQ3JDOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXVCVDs4QkFpRDBCLElBQUk7c0JBQTlCLEtBQUs7dUJBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFO2dCQUNFLFNBQVM7c0JBQW5DLEtBQUs7dUJBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFO2dCQUNmLE1BQU07c0JBQWYsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbIi8vIGxpYi9jb21wb25lbnRzL3RyZWUtbm9kZS1leHBhbmRlci90cmVlLW5vZGUtZXhwYW5kZXIuY29tcG9uZW50LnRzXHJcbmltcG9ydCB7XHJcbiAgQ29tcG9uZW50LFxyXG4gIElucHV0LFxyXG4gIE91dHB1dCxcclxuICBFdmVudEVtaXR0ZXIsXHJcbiAgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksXHJcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XHJcbmltcG9ydCB7IFRyZWVOb2RlIH0gZnJvbSAnLi4vLi4vY29yZS9tb2RlbHMvdHJlZS1ub2RlLm1vZGVsJztcclxuaW1wb3J0IHsgVHJlZU1vZGVsIH0gZnJvbSAnLi4vLi4vY29yZS9tb2RlbHMvdHJlZS5tb2RlbCc7XHJcblxyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogJ3RyZWUtbm9kZS1leHBhbmRlcicsXHJcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcclxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlXSxcclxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcclxuICB0ZW1wbGF0ZTogYFxyXG4gICAgPGRpdiBjbGFzcz1cInRyZWUtbm9kZS1leHBhbmRlclwiXHJcbiAgICAgICAgIFtjbGFzcy50cmVlLW5vZGUtZXhwYW5kZXItbG9hZGluZ109XCJub2RlLmlzTG9hZGluZygpXCJcclxuICAgICAgICAgKGNsaWNrKT1cIm9uRXhwYW5kZXJDbGljaygkZXZlbnQpXCI+XHJcbiAgICAgIEBpZiAobm9kZS5pc0xvYWRpbmcoKSkge1xyXG4gICAgICAgIDxzcGFuIGNsYXNzPVwiZXhwYW5kZXItc3Bpbm5lclwiPjwvc3Bhbj5cclxuICAgICAgfSBAZWxzZSBpZiAobm9kZS5oYXNDaGlsZHJlbikge1xyXG4gICAgICAgIDxzdmcgXHJcbiAgICAgICAgICBjbGFzcz1cImV4cGFuZGVyLWljb25cIlxyXG4gICAgICAgICAgW2NsYXNzLmV4cGFuZGVyLWljb24tZXhwYW5kZWRdPVwibm9kZS5pc0V4cGFuZGVkKClcIlxyXG4gICAgICAgICAgd2lkdGg9XCIxNlwiIFxyXG4gICAgICAgICAgaGVpZ2h0PVwiMTZcIiBcclxuICAgICAgICAgIHZpZXdCb3g9XCIwIDAgMTYgMTZcIj5cclxuICAgICAgICAgIDxwYXRoIGQ9XCJNNiA0bDQgNC00IDRcIiBcclxuICAgICAgICAgICAgICAgIGZpbGw9XCJub25lXCIgXHJcbiAgICAgICAgICAgICAgICBzdHJva2U9XCJjdXJyZW50Q29sb3JcIiBcclxuICAgICAgICAgICAgICAgIHN0cm9rZS13aWR0aD1cIjJcIiBcclxuICAgICAgICAgICAgICAgIHN0cm9rZS1saW5lY2FwPVwicm91bmRcIi8+XHJcbiAgICAgICAgPC9zdmc+XHJcbiAgICAgIH0gQGVsc2Uge1xyXG4gICAgICAgIDxzcGFuIGNsYXNzPVwiZXhwYW5kZXItcGxhY2Vob2xkZXJcIj48L3NwYW4+XHJcbiAgICAgIH1cclxuICAgIDwvZGl2PlxyXG4gIGAsXHJcbiAgc3R5bGVzOiBbYFxyXG4gICAgLnRyZWUtbm9kZS1leHBhbmRlciB7XHJcbiAgICAgIGRpc3BsYXk6IGZsZXg7XHJcbiAgICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XHJcbiAgICAgIGp1c3RpZnktY29udGVudDogY2VudGVyO1xyXG4gICAgICB3aWR0aDogMjBweDtcclxuICAgICAgaGVpZ2h0OiAyMHB4O1xyXG4gICAgICBjdXJzb3I6IHBvaW50ZXI7XHJcbiAgICAgIGJvcmRlci1yYWRpdXM6IDNweDtcclxuICAgICAgdHJhbnNpdGlvbjogYmFja2dyb3VuZC1jb2xvciAwLjE1cztcclxuICAgICAgZmxleC1zaHJpbms6IDA7XHJcbiAgICB9XHJcbiAgICBcclxuICAgIC50cmVlLW5vZGUtZXhwYW5kZXI6aG92ZXIge1xyXG4gICAgICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKDAsIDAsIDAsIDAuMDUpO1xyXG4gICAgfVxyXG4gICAgXHJcbiAgICAuZXhwYW5kZXItaWNvbiB7XHJcbiAgICAgIHRyYW5zaXRpb246IHRyYW5zZm9ybSAwLjJzIGVhc2U7XHJcbiAgICAgIHRyYW5zZm9ybTogcm90YXRlKDBkZWcpO1xyXG4gICAgfVxyXG4gICAgXHJcbiAgICAuZXhwYW5kZXItaWNvbi1leHBhbmRlZCB7XHJcbiAgICAgIHRyYW5zZm9ybTogcm90YXRlKDkwZGVnKTtcclxuICAgIH1cclxuICAgIFxyXG4gICAgLmV4cGFuZGVyLXBsYWNlaG9sZGVyIHtcclxuICAgICAgd2lkdGg6IDE2cHg7XHJcbiAgICAgIGhlaWdodDogMTZweDtcclxuICAgIH1cclxuICAgIFxyXG4gICAgLmV4cGFuZGVyLXNwaW5uZXIge1xyXG4gICAgICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7XHJcbiAgICAgIHdpZHRoOiAxNHB4O1xyXG4gICAgICBoZWlnaHQ6IDE0cHg7XHJcbiAgICAgIGJvcmRlcjogMnB4IHNvbGlkICNmM2YzZjM7XHJcbiAgICAgIGJvcmRlci10b3A6IDJweCBzb2xpZCAjMjE5NmYzO1xyXG4gICAgICBib3JkZXItcmFkaXVzOiA1MCU7XHJcbiAgICAgIGFuaW1hdGlvbjogc3BpbiAxcyBsaW5lYXIgaW5maW5pdGU7XHJcbiAgICB9XHJcbiAgICBcclxuICAgIEBrZXlmcmFtZXMgc3BpbiB7XHJcbiAgICAgIDAlIHsgdHJhbnNmb3JtOiByb3RhdGUoMGRlZyk7IH1cclxuICAgICAgMTAwJSB7IHRyYW5zZm9ybTogcm90YXRlKDM2MGRlZyk7IH1cclxuICAgIH1cclxuICBgXSxcclxufSlcclxuZXhwb3J0IGNsYXNzIFRyZWVOb2RlRXhwYW5kZXJDb21wb25lbnQge1xyXG4gIEBJbnB1dCh7IHJlcXVpcmVkOiB0cnVlIH0pIG5vZGUhOiBUcmVlTm9kZTtcclxuICBASW5wdXQoeyByZXF1aXJlZDogdHJ1ZSB9KSB0cmVlTW9kZWwhOiBUcmVlTW9kZWw7XHJcbiAgQE91dHB1dCgpIHRvZ2dsZSA9IG5ldyBFdmVudEVtaXR0ZXI8dm9pZD4oKTtcclxuICBcclxuICBvbkV4cGFuZGVyQ2xpY2soZXZlbnQ6IE1vdXNlRXZlbnQpOiB2b2lkIHtcclxuICAgIGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpO1xyXG4gICAgdGhpcy50b2dnbGUuZW1pdCgpO1xyXG4gIH1cclxufSJdfQ==
|