concepto-user-controls 0.0.10 → 0.0.15
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-tree/concepto-tree.component.mjs +39 -33
- package/esm2022/public-api.mjs +5 -1
- package/fesm2022/concepto-user-controls.mjs +38 -32
- package/fesm2022/concepto-user-controls.mjs.map +1 -1
- package/lib/concepto-tree/concepto-tree.component.d.ts +2 -3
- package/package.json +1 -1
- package/public-api.d.ts +4 -0
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { Component, Input, Output, EventEmitter, ViewChild, ChangeDetectionStrategy, signal } from '@angular/core';
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ViewChild, ChangeDetectionStrategy, signal, computed } from '@angular/core';
|
|
2
2
|
import { CommonModule } from '@angular/common';
|
|
3
3
|
import { FormsModule } from '@angular/forms';
|
|
4
4
|
import { TreeRootComponent } from './components/tree-root/tree-root.component';
|
|
@@ -20,50 +20,56 @@ export class ConceptoTreeComponent {
|
|
|
20
20
|
loadChildren = new EventEmitter();
|
|
21
21
|
treeEvent = new EventEmitter();
|
|
22
22
|
treeRoot;
|
|
23
|
+
// Estado con Signals
|
|
23
24
|
isDropdownOpen = signal(false);
|
|
24
|
-
searchTerm = '';
|
|
25
|
+
searchTerm = signal(''); // Cambiamos a Signal
|
|
25
26
|
selectedNode = signal(null);
|
|
26
|
-
//
|
|
27
|
+
// Almacén de todos los nodos (no necesita ser signal porque no cambia tras la carga inicial)
|
|
27
28
|
allNodes = [];
|
|
28
|
-
|
|
29
|
+
// Signal computada: Se actualiza sola cuando cambia searchTerm o allNodes es asignado
|
|
30
|
+
filteredNodes = computed(() => {
|
|
31
|
+
const term = this.searchTerm().toLowerCase();
|
|
32
|
+
if (!term)
|
|
33
|
+
return this.allNodes;
|
|
34
|
+
return this.allNodes.filter(node => String(node.data[this.displayField]).toLowerCase().includes(term));
|
|
35
|
+
});
|
|
29
36
|
get displayField() {
|
|
30
37
|
return this.options.displayField || 'name';
|
|
31
38
|
}
|
|
32
39
|
toggleDropdown() {
|
|
33
40
|
this.isDropdownOpen.update(v => !v);
|
|
34
41
|
if (this.isDropdownOpen()) {
|
|
35
|
-
this.searchTerm
|
|
36
|
-
this.filterNodes();
|
|
42
|
+
this.searchTerm.set(''); // Reseteamos el término al abrir
|
|
37
43
|
}
|
|
38
44
|
}
|
|
39
45
|
closeDropdown() {
|
|
40
46
|
this.isDropdownOpen.set(false);
|
|
41
47
|
}
|
|
42
|
-
filterNodes() {
|
|
43
|
-
if (!this.searchTerm) {
|
|
44
|
-
this.filteredNodes.set(this.allNodes);
|
|
45
|
-
}
|
|
46
|
-
else {
|
|
47
|
-
const term = this.searchTerm.toLowerCase();
|
|
48
|
-
this.filteredNodes.set(this.allNodes.filter(node => String(node.data[this.displayField]).toLowerCase().includes(term)));
|
|
49
|
-
}
|
|
50
|
-
}
|
|
51
48
|
selectNode(node) {
|
|
52
49
|
this.selectedNode.set(node);
|
|
53
50
|
this.isDropdownOpen.set(false);
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
51
|
+
if (this.treeRoot?.treeModel) {
|
|
52
|
+
// Usamos el API de la librería para activar
|
|
53
|
+
node.setActiveAndVisible();
|
|
54
|
+
// O si prefieres: this.treeRoot.treeModel.activateNode(node);
|
|
58
55
|
}
|
|
59
56
|
}
|
|
60
57
|
onTreeInitialized(event) {
|
|
61
58
|
this.initialized.emit(event);
|
|
62
|
-
//
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
59
|
+
// Mantenemos el setTimeout para evitar el error 'firstCreatePass'
|
|
60
|
+
setTimeout(() => {
|
|
61
|
+
if (event.treeModel) {
|
|
62
|
+
this.allNodes = event.treeModel.getAllNodes();
|
|
63
|
+
// Buscamos manualmente el nodo activo si existe
|
|
64
|
+
const activeNode = this.allNodes.find(n => n.isActive);
|
|
65
|
+
if (activeNode) {
|
|
66
|
+
this.selectedNode.set(activeNode);
|
|
67
|
+
}
|
|
68
|
+
// Forzamos una actualización inicial de la lista filtrada
|
|
69
|
+
// Al setear searchTerm de nuevo, disparas el computed
|
|
70
|
+
this.searchTerm.set('');
|
|
71
|
+
}
|
|
72
|
+
});
|
|
67
73
|
}
|
|
68
74
|
onNodeActivated(event) {
|
|
69
75
|
this.activate.emit(event);
|
|
@@ -79,7 +85,7 @@ export class ConceptoTreeComponent {
|
|
|
79
85
|
<div class="selected-option">
|
|
80
86
|
{{ selectedNode() ? selectedNode()?.data?.[displayField] : 'Buscar nodo...' }}
|
|
81
87
|
</div>
|
|
82
|
-
<div class="arrow"
|
|
88
|
+
<div class="arrow"><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></div>
|
|
83
89
|
</div>
|
|
84
90
|
|
|
85
91
|
<div class="backdrop" *ngIf="isDropdownOpen()" (click)="closeDropdown()"></div>
|
|
@@ -88,8 +94,8 @@ export class ConceptoTreeComponent {
|
|
|
88
94
|
<div class="search-input-wrapper" (click)="$event.stopPropagation()">
|
|
89
95
|
<input
|
|
90
96
|
type="text"
|
|
91
|
-
[
|
|
92
|
-
(ngModelChange)="
|
|
97
|
+
[ngModel]="searchTerm()"
|
|
98
|
+
(ngModelChange)="searchTerm.set($event)"
|
|
93
99
|
placeholder="Buscar..."
|
|
94
100
|
class="search-input"
|
|
95
101
|
#searchInput
|
|
@@ -127,7 +133,7 @@ export class ConceptoTreeComponent {
|
|
|
127
133
|
(treeEvent)="treeEvent.emit($event)"
|
|
128
134
|
></tree-root>
|
|
129
135
|
</div>
|
|
130
|
-
`, isInline: true, styles: [".concepto-tree-container{display:flex;flex-direction:column;height:100%;gap:10px}.search-bar-container{position:relative;width:100%;z-index:100}.custom-select{display:flex;justify-content:space-between;align-items:center;padding:8px 12px;border:1px solid #ccc;border-radius:4px;cursor:pointer;background:#
|
|
136
|
+
`, isInline: true, styles: [".concepto-tree-container{display:flex;flex-direction:column;height:100%;gap:10px}.search-bar-container{position:relative;width:100%;z-index:100}.custom-select{display:flex;justify-content:space-between;align-items:center;padding:8px 12px;border:1px solid #ccc;border-radius:4px;cursor:pointer;background:#e7e7e7;-webkit-user-select:none;user-select:none;transition:border-color .2s}.custom-select:hover{border-color:#888}.custom-select.open{border-color:#007bff;border-bottom-left-radius:0;border-bottom-right-radius:0}.backdrop{position:fixed;top:0;left:0;width:100%;height:100%;z-index:101;background:transparent}.dropdown-menu{position:absolute;top:100%;left:0;width:100%;background:#fff;border:1px solid #ccc;border-top:none;border-radius:0 0 4px 4px;box-shadow:0 4px 6px #0000001a;margin-top:0;max-height:300px;overflow-y:auto;z-index:102}.search-input-wrapper{padding:8px;position:sticky;top:0;background:#fff;border-bottom:1px solid #eee;z-index:103}.search-input{width:100%;padding:8px;border:1px solid #ddd;border-radius:4px;box-sizing:border-box;outline:none;background:#e7e7e7}.search-input:focus{border-color:#007bff}.options-list{max-height:250px;overflow-y:auto}.option-item{padding:8px 12px;cursor:pointer;transition:background-color .1s}.option-item:hover{background-color:#f5f5f5}.no-results{padding:12px;color:#888;font-style:italic;text-align:center}tree-root{flex:1;overflow:hidden}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: TreeRootComponent, selector: "tree-root", inputs: ["nodes", "options"], outputs: ["initialized", "toggleExpanded", "activate", "deactivate", "select", "deselect", "focus", "blur", "moveNode", "loadChildren", "treeEvent"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
131
137
|
}
|
|
132
138
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ConceptoTreeComponent, decorators: [{
|
|
133
139
|
type: Component,
|
|
@@ -138,7 +144,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
138
144
|
<div class="selected-option">
|
|
139
145
|
{{ selectedNode() ? selectedNode()?.data?.[displayField] : 'Buscar nodo...' }}
|
|
140
146
|
</div>
|
|
141
|
-
<div class="arrow"
|
|
147
|
+
<div class="arrow"><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></div>
|
|
142
148
|
</div>
|
|
143
149
|
|
|
144
150
|
<div class="backdrop" *ngIf="isDropdownOpen()" (click)="closeDropdown()"></div>
|
|
@@ -147,8 +153,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
147
153
|
<div class="search-input-wrapper" (click)="$event.stopPropagation()">
|
|
148
154
|
<input
|
|
149
155
|
type="text"
|
|
150
|
-
[
|
|
151
|
-
(ngModelChange)="
|
|
156
|
+
[ngModel]="searchTerm()"
|
|
157
|
+
(ngModelChange)="searchTerm.set($event)"
|
|
152
158
|
placeholder="Buscar..."
|
|
153
159
|
class="search-input"
|
|
154
160
|
#searchInput
|
|
@@ -186,7 +192,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
186
192
|
(treeEvent)="treeEvent.emit($event)"
|
|
187
193
|
></tree-root>
|
|
188
194
|
</div>
|
|
189
|
-
`, styles: [".concepto-tree-container{display:flex;flex-direction:column;height:100%;gap:10px}.search-bar-container{position:relative;width:100%;z-index:100}.custom-select{display:flex;justify-content:space-between;align-items:center;padding:8px 12px;border:1px solid #ccc;border-radius:4px;cursor:pointer;background:#
|
|
195
|
+
`, styles: [".concepto-tree-container{display:flex;flex-direction:column;height:100%;gap:10px}.search-bar-container{position:relative;width:100%;z-index:100}.custom-select{display:flex;justify-content:space-between;align-items:center;padding:8px 12px;border:1px solid #ccc;border-radius:4px;cursor:pointer;background:#e7e7e7;-webkit-user-select:none;user-select:none;transition:border-color .2s}.custom-select:hover{border-color:#888}.custom-select.open{border-color:#007bff;border-bottom-left-radius:0;border-bottom-right-radius:0}.backdrop{position:fixed;top:0;left:0;width:100%;height:100%;z-index:101;background:transparent}.dropdown-menu{position:absolute;top:100%;left:0;width:100%;background:#fff;border:1px solid #ccc;border-top:none;border-radius:0 0 4px 4px;box-shadow:0 4px 6px #0000001a;margin-top:0;max-height:300px;overflow-y:auto;z-index:102}.search-input-wrapper{padding:8px;position:sticky;top:0;background:#fff;border-bottom:1px solid #eee;z-index:103}.search-input{width:100%;padding:8px;border:1px solid #ddd;border-radius:4px;box-sizing:border-box;outline:none;background:#e7e7e7}.search-input:focus{border-color:#007bff}.options-list{max-height:250px;overflow-y:auto}.option-item{padding:8px 12px;cursor:pointer;transition:background-color .1s}.option-item:hover{background-color:#f5f5f5}.no-results{padding:12px;color:#888;font-style:italic;text-align:center}tree-root{flex:1;overflow:hidden}\n"] }]
|
|
190
196
|
}], propDecorators: { nodes: [{
|
|
191
197
|
type: Input
|
|
192
198
|
}], options: [{
|
|
@@ -217,4 +223,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
217
223
|
type: ViewChild,
|
|
218
224
|
args: [TreeRootComponent]
|
|
219
225
|
}] } });
|
|
220
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"concepto-tree.component.js","sourceRoot":"","sources":["../../../../../projects/concepto-user-controls/src/lib/concepto-tree/concepto-tree.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, ViewChild, ChangeDetectionStrategy, signal } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { TreeRootComponent } from './components/tree-root/tree-root.component';\r\nimport { TreeOptions } from './core/models/tree-options.model';\r\nimport { TreeEvent } from './core/models/tree-events.model';\r\nimport { TreeNode } from './core/models/tree-node.model';\r\n\r\n@Component({\r\n  selector: 'lib-concepto-tree',\r\n  standalone: true,\r\n  imports: [CommonModule, FormsModule, TreeRootComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <div class=\"concepto-tree-container\">\r\n      <div class=\"search-bar-container\">\r\n        <div class=\"custom-select\" [class.open]=\"isDropdownOpen()\" (click)=\"toggleDropdown()\">\r\n           <div class=\"selected-option\">\r\n             {{ selectedNode() ? selectedNode()?.data?.[displayField] : 'Buscar nodo...' }}\r\n           </div>\r\n           <div class=\"arrow\">▼</div>\r\n        </div>\r\n        \r\n        <div class=\"backdrop\" *ngIf=\"isDropdownOpen()\" (click)=\"closeDropdown()\"></div>\r\n\r\n        <div class=\"dropdown-menu\" *ngIf=\"isDropdownOpen()\">\r\n           <div class=\"search-input-wrapper\" (click)=\"$event.stopPropagation()\">\r\n             <input \r\n               type=\"text\" \r\n               [(ngModel)]=\"searchTerm\" \r\n               (ngModelChange)=\"filterNodes()\"\r\n               placeholder=\"Buscar...\"\r\n               class=\"search-input\"\r\n               #searchInput\r\n               autofocus\r\n             >\r\n           </div>\r\n           <div class=\"options-list\">\r\n             <div \r\n               *ngFor=\"let node of filteredNodes()\" \r\n               class=\"option-item\" \r\n               (click)=\"selectNode(node); $event.stopPropagation()\"\r\n             >\r\n               {{ node.data[displayField] }}\r\n             </div>\r\n             <div *ngIf=\"filteredNodes().length === 0\" class=\"no-results\">\r\n               No se encontraron resultados\r\n             </div>\r\n           </div>\r\n        </div>\r\n      </div>\r\n\r\n      <tree-root\r\n        [nodes]=\"nodes\"\r\n        [options]=\"options\"\r\n        (initialized)=\"onTreeInitialized($event)\"\r\n        (toggleExpanded)=\"toggleExpanded.emit($event)\"\r\n        (activate)=\"onNodeActivated($event)\"\r\n        (deactivate)=\"deactivate.emit($event)\"\r\n        (select)=\"select.emit($event)\"\r\n        (deselect)=\"deselect.emit($event)\"\r\n        (focus)=\"focus.emit($event)\"\r\n        (blur)=\"blur.emit($event)\"\r\n        (moveNode)=\"moveNode.emit($event)\"\r\n        (loadChildren)=\"loadChildren.emit($event)\"\r\n        (treeEvent)=\"treeEvent.emit($event)\"\r\n      ></tree-root>\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    .concepto-tree-container {\r\n      display: flex;\r\n      flex-direction: column;\r\n      height: 100%;\r\n      gap: 10px;\r\n    }\r\n    \r\n    .search-bar-container {\r\n      position: relative;\r\n      width: 100%;\r\n      z-index: 100;\r\n    }\r\n\r\n    .custom-select {\r\n      display: flex;\r\n      justify-content: space-between;\r\n      align-items: center;\r\n      padding: 8px 12px;\r\n      border: 1px solid #ccc;\r\n      border-radius: 4px;\r\n      cursor: pointer;\r\n      background: white;\r\n      user-select: none;\r\n      transition: border-color 0.2s;\r\n    }\r\n\r\n    .custom-select:hover {\r\n      border-color: #888;\r\n    }\r\n    \r\n    .custom-select.open {\r\n      border-color: #007bff;\r\n      border-bottom-left-radius: 0;\r\n      border-bottom-right-radius: 0;\r\n    }\r\n\r\n    .backdrop {\r\n      position: fixed;\r\n      top: 0;\r\n      left: 0;\r\n      width: 100%;\r\n      height: 100%;\r\n      z-index: 101;\r\n      background: transparent;\r\n    }\r\n\r\n    .dropdown-menu {\r\n      position: absolute;\r\n      top: 100%;\r\n      left: 0;\r\n      width: 100%;\r\n      background: white;\r\n      border: 1px solid #ccc;\r\n      border-top: none;\r\n      border-radius: 0 0 4px 4px;\r\n      box-shadow: 0 4px 6px rgba(0,0,0,0.1);\r\n      margin-top: 0;\r\n      max-height: 300px;\r\n      overflow-y: auto;\r\n      z-index: 102;\r\n    }\r\n\r\n    .search-input-wrapper {\r\n      padding: 8px;\r\n      position: sticky;\r\n      top: 0;\r\n      background: white;\r\n      border-bottom: 1px solid #eee;\r\n      z-index: 103;\r\n    }\r\n\r\n    .search-input {\r\n      width: 100%;\r\n      padding: 8px;\r\n      border: 1px solid #ddd;\r\n      border-radius: 4px;\r\n      box-sizing: border-box;\r\n      outline: none;\r\n    }\r\n    \r\n    .search-input:focus {\r\n      border-color: #007bff;\r\n    }\r\n\r\n    .options-list {\r\n      max-height: 250px;\r\n      overflow-y: auto;\r\n    }\r\n\r\n    .option-item {\r\n      padding: 8px 12px;\r\n      cursor: pointer;\r\n      transition: background-color 0.1s;\r\n    }\r\n\r\n    .option-item:hover {\r\n      background-color: #f5f5f5;\r\n    }\r\n\r\n    .no-results {\r\n      padding: 12px;\r\n      color: #888;\r\n      font-style: italic;\r\n      text-align: center;\r\n    }\r\n    \r\n    tree-root {\r\n      flex: 1;\r\n      overflow: hidden;\r\n    }\r\n  `]\r\n})\r\nexport class ConceptoTreeComponent {\r\n  @Input() nodes: any[] = [];\r\n  @Input() options: Partial<TreeOptions> = {};\r\n\r\n  @Output() initialized = new EventEmitter<TreeEvent>();\r\n  @Output() toggleExpanded = new EventEmitter<TreeEvent>();\r\n  @Output() activate = new EventEmitter<TreeEvent>();\r\n  @Output() deactivate = new EventEmitter<TreeEvent>();\r\n  @Output() select = new EventEmitter<TreeEvent>();\r\n  @Output() deselect = new EventEmitter<TreeEvent>();\r\n  @Output() focus = new EventEmitter<TreeEvent>();\r\n  @Output() blur = new EventEmitter<TreeEvent>();\r\n  @Output() moveNode = new EventEmitter<TreeEvent>();\r\n  @Output() loadChildren = new EventEmitter<TreeEvent>();\r\n  @Output() treeEvent = new EventEmitter<TreeEvent>();\r\n\r\n  @ViewChild(TreeRootComponent) treeRoot!: TreeRootComponent;\r\n\r\n  isDropdownOpen = signal(false);\r\n  searchTerm = '';\r\n  selectedNode = signal<TreeNode | null>(null);\r\n\r\n  // All nodes flattened\r\n  private allNodes: TreeNode[] = [];\r\n  filteredNodes = signal<TreeNode[]>([]);\r\n\r\n  get displayField(): string {\r\n    return this.options.displayField || 'name';\r\n  }\r\n\r\n  toggleDropdown() {\r\n    this.isDropdownOpen.update(v => !v);\r\n    if (this.isDropdownOpen()) {\r\n      this.searchTerm = '';\r\n      this.filterNodes();\r\n    }\r\n  }\r\n\r\n  closeDropdown() {\r\n    this.isDropdownOpen.set(false);\r\n  }\r\n\r\n  filterNodes() {\r\n    if (!this.searchTerm) {\r\n      this.filteredNodes.set(this.allNodes);\r\n    } else {\r\n      const term = this.searchTerm.toLowerCase();\r\n      this.filteredNodes.set(\r\n        this.allNodes.filter(node =>\r\n          String(node.data[this.displayField]).toLowerCase().includes(term)\r\n        )\r\n      );\r\n    }\r\n  }\r\n\r\n  selectNode(node: TreeNode) {\r\n    this.selectedNode.set(node);\r\n    this.isDropdownOpen.set(false);\r\n\r\n    // Activate in tree\r\n    if (this.treeRoot && this.treeRoot.treeModel) {\r\n      this.treeRoot.treeModel.activateNode(node);\r\n      node.ensureVisible();\r\n    }\r\n  }\r\n\r\n  onTreeInitialized(event: TreeEvent) {\r\n    this.initialized.emit(event);\r\n    // Get all nodes for the dropdown\r\n    if (event.treeModel) {\r\n      this.allNodes = event.treeModel.getAllNodes();\r\n      this.filteredNodes.set(this.allNodes);\r\n    }\r\n  }\r\n\r\n  onNodeActivated(event: TreeEvent) {\r\n    this.activate.emit(event);\r\n    if (event.node) {\r\n      this.selectedNode.set(event.node);\r\n    }\r\n  }\r\n}\r\n"]}
|
|
226
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"concepto-tree.component.js","sourceRoot":"","sources":["../../../../../projects/concepto-user-controls/src/lib/concepto-tree/concepto-tree.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, ViewChild, ChangeDetectionStrategy, signal, computed } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { TreeRootComponent } from './components/tree-root/tree-root.component';\r\nimport { TreeOptions } from './core/models/tree-options.model';\r\nimport { TreeEvent } from './core/models/tree-events.model';\r\nimport { TreeNode } from './core/models/tree-node.model';\r\n\r\n@Component({\r\n  selector: 'lib-concepto-tree',\r\n  standalone: true,\r\n  imports: [CommonModule, FormsModule, TreeRootComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <div class=\"concepto-tree-container\">\r\n      <div class=\"search-bar-container\">\r\n        <div class=\"custom-select\" [class.open]=\"isDropdownOpen()\" (click)=\"toggleDropdown()\">\r\n           <div class=\"selected-option\">\r\n             {{ selectedNode() ? selectedNode()?.data?.[displayField] : 'Buscar nodo...' }}\r\n           </div>\r\n           <div class=\"arrow\"><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></div>\r\n        </div>\r\n        \r\n        <div class=\"backdrop\" *ngIf=\"isDropdownOpen()\" (click)=\"closeDropdown()\"></div>\r\n\r\n        <div class=\"dropdown-menu\" *ngIf=\"isDropdownOpen()\">\r\n           <div class=\"search-input-wrapper\" (click)=\"$event.stopPropagation()\">\r\n             <input \r\n               type=\"text\" \r\n               [ngModel]=\"searchTerm()\"\r\n               (ngModelChange)=\"searchTerm.set($event)\" \r\n               placeholder=\"Buscar...\"\r\n               class=\"search-input\"\r\n               #searchInput\r\n               autofocus\r\n             >\r\n           </div>\r\n           <div class=\"options-list\">\r\n             <div \r\n               *ngFor=\"let node of filteredNodes()\" \r\n               class=\"option-item\" \r\n               (click)=\"selectNode(node); $event.stopPropagation()\"\r\n             >\r\n               {{ node.data[displayField] }}\r\n             </div>\r\n             <div *ngIf=\"filteredNodes().length === 0\" class=\"no-results\">\r\n               No se encontraron resultados\r\n             </div>\r\n           </div>\r\n        </div>\r\n      </div>\r\n\r\n      <tree-root\r\n        [nodes]=\"nodes\"\r\n        [options]=\"options\"\r\n        (initialized)=\"onTreeInitialized($event)\"\r\n        (toggleExpanded)=\"toggleExpanded.emit($event)\"\r\n        (activate)=\"onNodeActivated($event)\"\r\n        (deactivate)=\"deactivate.emit($event)\"\r\n        (select)=\"select.emit($event)\"\r\n        (deselect)=\"deselect.emit($event)\"\r\n        (focus)=\"focus.emit($event)\"\r\n        (blur)=\"blur.emit($event)\"\r\n        (moveNode)=\"moveNode.emit($event)\"\r\n        (loadChildren)=\"loadChildren.emit($event)\"\r\n        (treeEvent)=\"treeEvent.emit($event)\"\r\n      ></tree-root>\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    .concepto-tree-container {\r\n      display: flex;\r\n      flex-direction: column;\r\n      height: 100%;\r\n      gap: 10px;\r\n    }\r\n    \r\n    .search-bar-container {\r\n      position: relative;\r\n      width: 100%;\r\n      z-index: 100;\r\n    }\r\n\r\n    .custom-select {\r\n      display: flex;\r\n      justify-content: space-between;\r\n      align-items: center;\r\n      padding: 8px 12px;\r\n      border: 1px solid #ccc;\r\n      border-radius: 4px;\r\n      cursor: pointer;\r\n      background: #e7e7e7;\r\n      user-select: none;\r\n      transition: border-color 0.2s;\r\n    }\r\n\r\n    .custom-select:hover {\r\n      border-color: #888;\r\n    }\r\n    \r\n    .custom-select.open {\r\n      border-color: #007bff;\r\n      border-bottom-left-radius: 0;\r\n      border-bottom-right-radius: 0;\r\n    }\r\n\r\n    .backdrop {\r\n      position: fixed;\r\n      top: 0;\r\n      left: 0;\r\n      width: 100%;\r\n      height: 100%;\r\n      z-index: 101;\r\n      background: transparent;\r\n    }\r\n\r\n    .dropdown-menu {\r\n      position: absolute;\r\n      top: 100%;\r\n      left: 0;\r\n      width: 100%;\r\n      background: white;\r\n      border: 1px solid #ccc;\r\n      border-top: none;\r\n      border-radius: 0 0 4px 4px;\r\n      box-shadow: 0 4px 6px rgba(0,0,0,0.1);\r\n      margin-top: 0;\r\n      max-height: 300px;\r\n      overflow-y: auto;\r\n      z-index: 102;\r\n    }\r\n\r\n    .search-input-wrapper {\r\n      padding: 8px;\r\n      position: sticky;\r\n      top: 0;\r\n      background: white;\r\n      border-bottom: 1px solid #eee;\r\n      z-index: 103;\r\n    }\r\n\r\n    .search-input {\r\n      width: 100%;\r\n      padding: 8px;\r\n      border: 1px solid #ddd;\r\n      border-radius: 4px;\r\n      box-sizing: border-box;\r\n      outline: none;\r\n      background: #e7e7e7;\r\n    }\r\n    \r\n    .search-input:focus {\r\n      border-color: #007bff;\r\n    }\r\n\r\n    .options-list {\r\n      max-height: 250px;\r\n      overflow-y: auto;\r\n    }\r\n\r\n    .option-item {\r\n      padding: 8px 12px;\r\n      cursor: pointer;\r\n      transition: background-color 0.1s;\r\n    }\r\n\r\n    .option-item:hover {\r\n      background-color: #f5f5f5;\r\n    }\r\n\r\n    .no-results {\r\n      padding: 12px;\r\n      color: #888;\r\n      font-style: italic;\r\n      text-align: center;\r\n    }\r\n    \r\n    tree-root {\r\n      flex: 1;\r\n      overflow: hidden;\r\n    }\r\n  `]\r\n})\r\nexport class ConceptoTreeComponent {\r\n  @Input() nodes: any[] = [];\r\n  @Input() options: Partial<TreeOptions> = {};\r\n\r\n  @Output() initialized = new EventEmitter<TreeEvent>();\r\n  @Output() toggleExpanded = new EventEmitter<TreeEvent>();\r\n  @Output() activate = new EventEmitter<TreeEvent>();\r\n  @Output() deactivate = new EventEmitter<TreeEvent>();\r\n  @Output() select = new EventEmitter<TreeEvent>();\r\n  @Output() deselect = new EventEmitter<TreeEvent>();\r\n  @Output() focus = new EventEmitter<TreeEvent>();\r\n  @Output() blur = new EventEmitter<TreeEvent>();\r\n  @Output() moveNode = new EventEmitter<TreeEvent>();\r\n  @Output() loadChildren = new EventEmitter<TreeEvent>();\r\n  @Output() treeEvent = new EventEmitter<TreeEvent>();\r\n\r\n  @ViewChild(TreeRootComponent) treeRoot!: TreeRootComponent;\r\n\r\n  // Estado con Signals\r\n  isDropdownOpen = signal(false);\r\n  searchTerm = signal(''); // Cambiamos a Signal\r\n  selectedNode = signal<TreeNode | null>(null);\r\n  \r\n  // Almacén de todos los nodos (no necesita ser signal porque no cambia tras la carga inicial)\r\n  private allNodes: TreeNode[] = [];\r\n\r\n  // Signal computada: Se actualiza sola cuando cambia searchTerm o allNodes es asignado\r\n  filteredNodes = computed(() => {\r\n    const term = this.searchTerm().toLowerCase();\r\n    if (!term) return this.allNodes;\r\n    \r\n    return this.allNodes.filter(node =>\r\n      String(node.data[this.displayField]).toLowerCase().includes(term)\r\n    );\r\n  });\r\n\r\n  get displayField(): string {\r\n    return this.options.displayField || 'name';\r\n  }\r\n\r\n  toggleDropdown() {\r\n    this.isDropdownOpen.update(v => !v);\r\n    if (this.isDropdownOpen()) {\r\n      this.searchTerm.set(''); // Reseteamos el término al abrir\r\n    }\r\n  }\r\n\r\n  closeDropdown() {\r\n    this.isDropdownOpen.set(false);\r\n  }\r\n\r\n  selectNode(node: TreeNode) {\r\n    this.selectedNode.set(node);\r\n    this.isDropdownOpen.set(false);\r\n\r\n    if (this.treeRoot?.treeModel) {\r\n      // Usamos el API de la librería para activar\r\n      node.setActiveAndVisible(); \r\n      // O si prefieres: this.treeRoot.treeModel.activateNode(node);\r\n    }\r\n  }\r\n\r\n  onTreeInitialized(event: TreeEvent) {\r\n    this.initialized.emit(event);\r\n  \r\n    // Mantenemos el setTimeout para evitar el error 'firstCreatePass'\r\n    setTimeout(() => {\r\n      if (event.treeModel) {\r\n        this.allNodes = event.treeModel.getAllNodes();\r\n        \r\n        // Buscamos manualmente el nodo activo si existe\r\n        const activeNode = this.allNodes.find(n => n.isActive);\r\n        if (activeNode) {\r\n          this.selectedNode.set(activeNode);\r\n        }\r\n        \r\n        // Forzamos una actualización inicial de la lista filtrada\r\n        // Al setear searchTerm de nuevo, disparas el computed\r\n        this.searchTerm.set('');\r\n      }\r\n    });\r\n  }\r\n\r\n  onNodeActivated(event: TreeEvent) {\r\n    this.activate.emit(event);\r\n    if (event.node) {\r\n      this.selectedNode.set(event.node);\r\n    }\r\n  }\r\n}\r\n"]}
|
package/esm2022/public-api.mjs
CHANGED
|
@@ -6,7 +6,11 @@ export * from './lib/concepto-user-controls.component';
|
|
|
6
6
|
export * from './lib/concepto-message/concepto-message.component';
|
|
7
7
|
export * from './lib/concepto-context-menu/concepto-context-menu.component';
|
|
8
8
|
export * from './lib/concepto-tree/concepto-tree.component';
|
|
9
|
+
export * from './lib/concepto-tree/core/models/tree.model';
|
|
10
|
+
export * from './lib/concepto-tree/core/models/tree-node.model';
|
|
11
|
+
export * from './lib/concepto-tree/core/models/tree-options.model';
|
|
12
|
+
export * from './lib/concepto-tree/core/models/tree-events.model';
|
|
9
13
|
export * from './lib/entity-comparison/components/entity-comparison.component';
|
|
10
14
|
export * from './lib/entity-comparison/core/services/entity-comparison.service';
|
|
11
15
|
export * from './lib/concepto-input/concepto-input.component';
|
|
12
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2NvbmNlcHRvLXVzZXItY29udHJvbHMvc3JjL3B1YmxpYy1hcGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLHNDQUFzQyxDQUFDO0FBQ3JELGNBQWMsd0NBQXdDLENBQUM7QUFDdkQsY0FBYyxtREFBbUQsQ0FBQztBQUNsRSxjQUFjLDZEQUE2RCxDQUFDO0FBQzVFLGNBQWMsNkNBQTZDLENBQUM7QUFDNUQsY0FBYyw0Q0FBNEMsQ0FBQztBQUMzRCxjQUFjLGlEQUFpRCxDQUFDO0FBQ2hFLGNBQWMsb0RBQW9ELENBQUM7QUFDbkUsY0FBYyxtREFBbUQsQ0FBQztBQUNsRSxjQUFjLGdFQUFnRSxDQUFDO0FBQy9FLGNBQWMsaUVBQWlFLENBQUM7QUFDaEYsY0FBYywrQ0FBK0MsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXHJcbiAqIFB1YmxpYyBBUEkgU3VyZmFjZSBvZiBjb25jZXB0by1tZXNzYWdlXHJcbiAqL1xyXG5cclxuZXhwb3J0ICogZnJvbSAnLi9saWIvY29uY2VwdG8tdXNlci1jb250cm9scy5zZXJ2aWNlJztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvY29uY2VwdG8tdXNlci1jb250cm9scy5jb21wb25lbnQnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9jb25jZXB0by1tZXNzYWdlL2NvbmNlcHRvLW1lc3NhZ2UuY29tcG9uZW50JztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvY29uY2VwdG8tY29udGV4dC1tZW51L2NvbmNlcHRvLWNvbnRleHQtbWVudS5jb21wb25lbnQnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9jb25jZXB0by10cmVlL2NvbmNlcHRvLXRyZWUuY29tcG9uZW50JztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvY29uY2VwdG8tdHJlZS9jb3JlL21vZGVscy90cmVlLm1vZGVsJztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvY29uY2VwdG8tdHJlZS9jb3JlL21vZGVscy90cmVlLW5vZGUubW9kZWwnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9jb25jZXB0by10cmVlL2NvcmUvbW9kZWxzL3RyZWUtb3B0aW9ucy5tb2RlbCc7XHJcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NvbmNlcHRvLXRyZWUvY29yZS9tb2RlbHMvdHJlZS1ldmVudHMubW9kZWwnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9lbnRpdHktY29tcGFyaXNvbi9jb21wb25lbnRzL2VudGl0eS1jb21wYXJpc29uLmNvbXBvbmVudCc7XHJcbmV4cG9ydCAqIGZyb20gJy4vbGliL2VudGl0eS1jb21wYXJpc29uL2NvcmUvc2VydmljZXMvZW50aXR5LWNvbXBhcmlzb24uc2VydmljZSc7XHJcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NvbmNlcHRvLWlucHV0L2NvbmNlcHRvLWlucHV0LmNvbXBvbmVudCc7Il19
|
|
@@ -1755,50 +1755,56 @@ class ConceptoTreeComponent {
|
|
|
1755
1755
|
loadChildren = new EventEmitter();
|
|
1756
1756
|
treeEvent = new EventEmitter();
|
|
1757
1757
|
treeRoot;
|
|
1758
|
+
// Estado con Signals
|
|
1758
1759
|
isDropdownOpen = signal(false);
|
|
1759
|
-
searchTerm = '';
|
|
1760
|
+
searchTerm = signal(''); // Cambiamos a Signal
|
|
1760
1761
|
selectedNode = signal(null);
|
|
1761
|
-
//
|
|
1762
|
+
// Almacén de todos los nodos (no necesita ser signal porque no cambia tras la carga inicial)
|
|
1762
1763
|
allNodes = [];
|
|
1763
|
-
|
|
1764
|
+
// Signal computada: Se actualiza sola cuando cambia searchTerm o allNodes es asignado
|
|
1765
|
+
filteredNodes = computed(() => {
|
|
1766
|
+
const term = this.searchTerm().toLowerCase();
|
|
1767
|
+
if (!term)
|
|
1768
|
+
return this.allNodes;
|
|
1769
|
+
return this.allNodes.filter(node => String(node.data[this.displayField]).toLowerCase().includes(term));
|
|
1770
|
+
});
|
|
1764
1771
|
get displayField() {
|
|
1765
1772
|
return this.options.displayField || 'name';
|
|
1766
1773
|
}
|
|
1767
1774
|
toggleDropdown() {
|
|
1768
1775
|
this.isDropdownOpen.update(v => !v);
|
|
1769
1776
|
if (this.isDropdownOpen()) {
|
|
1770
|
-
this.searchTerm
|
|
1771
|
-
this.filterNodes();
|
|
1777
|
+
this.searchTerm.set(''); // Reseteamos el término al abrir
|
|
1772
1778
|
}
|
|
1773
1779
|
}
|
|
1774
1780
|
closeDropdown() {
|
|
1775
1781
|
this.isDropdownOpen.set(false);
|
|
1776
1782
|
}
|
|
1777
|
-
filterNodes() {
|
|
1778
|
-
if (!this.searchTerm) {
|
|
1779
|
-
this.filteredNodes.set(this.allNodes);
|
|
1780
|
-
}
|
|
1781
|
-
else {
|
|
1782
|
-
const term = this.searchTerm.toLowerCase();
|
|
1783
|
-
this.filteredNodes.set(this.allNodes.filter(node => String(node.data[this.displayField]).toLowerCase().includes(term)));
|
|
1784
|
-
}
|
|
1785
|
-
}
|
|
1786
1783
|
selectNode(node) {
|
|
1787
1784
|
this.selectedNode.set(node);
|
|
1788
1785
|
this.isDropdownOpen.set(false);
|
|
1789
|
-
|
|
1790
|
-
|
|
1791
|
-
|
|
1792
|
-
|
|
1786
|
+
if (this.treeRoot?.treeModel) {
|
|
1787
|
+
// Usamos el API de la librería para activar
|
|
1788
|
+
node.setActiveAndVisible();
|
|
1789
|
+
// O si prefieres: this.treeRoot.treeModel.activateNode(node);
|
|
1793
1790
|
}
|
|
1794
1791
|
}
|
|
1795
1792
|
onTreeInitialized(event) {
|
|
1796
1793
|
this.initialized.emit(event);
|
|
1797
|
-
//
|
|
1798
|
-
|
|
1799
|
-
|
|
1800
|
-
|
|
1801
|
-
|
|
1794
|
+
// Mantenemos el setTimeout para evitar el error 'firstCreatePass'
|
|
1795
|
+
setTimeout(() => {
|
|
1796
|
+
if (event.treeModel) {
|
|
1797
|
+
this.allNodes = event.treeModel.getAllNodes();
|
|
1798
|
+
// Buscamos manualmente el nodo activo si existe
|
|
1799
|
+
const activeNode = this.allNodes.find(n => n.isActive);
|
|
1800
|
+
if (activeNode) {
|
|
1801
|
+
this.selectedNode.set(activeNode);
|
|
1802
|
+
}
|
|
1803
|
+
// Forzamos una actualización inicial de la lista filtrada
|
|
1804
|
+
// Al setear searchTerm de nuevo, disparas el computed
|
|
1805
|
+
this.searchTerm.set('');
|
|
1806
|
+
}
|
|
1807
|
+
});
|
|
1802
1808
|
}
|
|
1803
1809
|
onNodeActivated(event) {
|
|
1804
1810
|
this.activate.emit(event);
|
|
@@ -1814,7 +1820,7 @@ class ConceptoTreeComponent {
|
|
|
1814
1820
|
<div class="selected-option">
|
|
1815
1821
|
{{ selectedNode() ? selectedNode()?.data?.[displayField] : 'Buscar nodo...' }}
|
|
1816
1822
|
</div>
|
|
1817
|
-
<div class="arrow"
|
|
1823
|
+
<div class="arrow"><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></div>
|
|
1818
1824
|
</div>
|
|
1819
1825
|
|
|
1820
1826
|
<div class="backdrop" *ngIf="isDropdownOpen()" (click)="closeDropdown()"></div>
|
|
@@ -1823,8 +1829,8 @@ class ConceptoTreeComponent {
|
|
|
1823
1829
|
<div class="search-input-wrapper" (click)="$event.stopPropagation()">
|
|
1824
1830
|
<input
|
|
1825
1831
|
type="text"
|
|
1826
|
-
[
|
|
1827
|
-
(ngModelChange)="
|
|
1832
|
+
[ngModel]="searchTerm()"
|
|
1833
|
+
(ngModelChange)="searchTerm.set($event)"
|
|
1828
1834
|
placeholder="Buscar..."
|
|
1829
1835
|
class="search-input"
|
|
1830
1836
|
#searchInput
|
|
@@ -1862,7 +1868,7 @@ class ConceptoTreeComponent {
|
|
|
1862
1868
|
(treeEvent)="treeEvent.emit($event)"
|
|
1863
1869
|
></tree-root>
|
|
1864
1870
|
</div>
|
|
1865
|
-
`, isInline: true, styles: [".concepto-tree-container{display:flex;flex-direction:column;height:100%;gap:10px}.search-bar-container{position:relative;width:100%;z-index:100}.custom-select{display:flex;justify-content:space-between;align-items:center;padding:8px 12px;border:1px solid #ccc;border-radius:4px;cursor:pointer;background:#
|
|
1871
|
+
`, isInline: true, styles: [".concepto-tree-container{display:flex;flex-direction:column;height:100%;gap:10px}.search-bar-container{position:relative;width:100%;z-index:100}.custom-select{display:flex;justify-content:space-between;align-items:center;padding:8px 12px;border:1px solid #ccc;border-radius:4px;cursor:pointer;background:#e7e7e7;-webkit-user-select:none;user-select:none;transition:border-color .2s}.custom-select:hover{border-color:#888}.custom-select.open{border-color:#007bff;border-bottom-left-radius:0;border-bottom-right-radius:0}.backdrop{position:fixed;top:0;left:0;width:100%;height:100%;z-index:101;background:transparent}.dropdown-menu{position:absolute;top:100%;left:0;width:100%;background:#fff;border:1px solid #ccc;border-top:none;border-radius:0 0 4px 4px;box-shadow:0 4px 6px #0000001a;margin-top:0;max-height:300px;overflow-y:auto;z-index:102}.search-input-wrapper{padding:8px;position:sticky;top:0;background:#fff;border-bottom:1px solid #eee;z-index:103}.search-input{width:100%;padding:8px;border:1px solid #ddd;border-radius:4px;box-sizing:border-box;outline:none;background:#e7e7e7}.search-input:focus{border-color:#007bff}.options-list{max-height:250px;overflow-y:auto}.option-item{padding:8px 12px;cursor:pointer;transition:background-color .1s}.option-item:hover{background-color:#f5f5f5}.no-results{padding:12px;color:#888;font-style:italic;text-align:center}tree-root{flex:1;overflow:hidden}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: TreeRootComponent, selector: "tree-root", inputs: ["nodes", "options"], outputs: ["initialized", "toggleExpanded", "activate", "deactivate", "select", "deselect", "focus", "blur", "moveNode", "loadChildren", "treeEvent"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
1866
1872
|
}
|
|
1867
1873
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ConceptoTreeComponent, decorators: [{
|
|
1868
1874
|
type: Component,
|
|
@@ -1873,7 +1879,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
1873
1879
|
<div class="selected-option">
|
|
1874
1880
|
{{ selectedNode() ? selectedNode()?.data?.[displayField] : 'Buscar nodo...' }}
|
|
1875
1881
|
</div>
|
|
1876
|
-
<div class="arrow"
|
|
1882
|
+
<div class="arrow"><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></div>
|
|
1877
1883
|
</div>
|
|
1878
1884
|
|
|
1879
1885
|
<div class="backdrop" *ngIf="isDropdownOpen()" (click)="closeDropdown()"></div>
|
|
@@ -1882,8 +1888,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
1882
1888
|
<div class="search-input-wrapper" (click)="$event.stopPropagation()">
|
|
1883
1889
|
<input
|
|
1884
1890
|
type="text"
|
|
1885
|
-
[
|
|
1886
|
-
(ngModelChange)="
|
|
1891
|
+
[ngModel]="searchTerm()"
|
|
1892
|
+
(ngModelChange)="searchTerm.set($event)"
|
|
1887
1893
|
placeholder="Buscar..."
|
|
1888
1894
|
class="search-input"
|
|
1889
1895
|
#searchInput
|
|
@@ -1921,7 +1927,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
1921
1927
|
(treeEvent)="treeEvent.emit($event)"
|
|
1922
1928
|
></tree-root>
|
|
1923
1929
|
</div>
|
|
1924
|
-
`, styles: [".concepto-tree-container{display:flex;flex-direction:column;height:100%;gap:10px}.search-bar-container{position:relative;width:100%;z-index:100}.custom-select{display:flex;justify-content:space-between;align-items:center;padding:8px 12px;border:1px solid #ccc;border-radius:4px;cursor:pointer;background:#
|
|
1930
|
+
`, styles: [".concepto-tree-container{display:flex;flex-direction:column;height:100%;gap:10px}.search-bar-container{position:relative;width:100%;z-index:100}.custom-select{display:flex;justify-content:space-between;align-items:center;padding:8px 12px;border:1px solid #ccc;border-radius:4px;cursor:pointer;background:#e7e7e7;-webkit-user-select:none;user-select:none;transition:border-color .2s}.custom-select:hover{border-color:#888}.custom-select.open{border-color:#007bff;border-bottom-left-radius:0;border-bottom-right-radius:0}.backdrop{position:fixed;top:0;left:0;width:100%;height:100%;z-index:101;background:transparent}.dropdown-menu{position:absolute;top:100%;left:0;width:100%;background:#fff;border:1px solid #ccc;border-top:none;border-radius:0 0 4px 4px;box-shadow:0 4px 6px #0000001a;margin-top:0;max-height:300px;overflow-y:auto;z-index:102}.search-input-wrapper{padding:8px;position:sticky;top:0;background:#fff;border-bottom:1px solid #eee;z-index:103}.search-input{width:100%;padding:8px;border:1px solid #ddd;border-radius:4px;box-sizing:border-box;outline:none;background:#e7e7e7}.search-input:focus{border-color:#007bff}.options-list{max-height:250px;overflow-y:auto}.option-item{padding:8px 12px;cursor:pointer;transition:background-color .1s}.option-item:hover{background-color:#f5f5f5}.no-results{padding:12px;color:#888;font-style:italic;text-align:center}tree-root{flex:1;overflow:hidden}\n"] }]
|
|
1925
1931
|
}], propDecorators: { nodes: [{
|
|
1926
1932
|
type: Input
|
|
1927
1933
|
}], options: [{
|
|
@@ -2300,5 +2306,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
2300
2306
|
* Generated bundle index. Do not edit.
|
|
2301
2307
|
*/
|
|
2302
2308
|
|
|
2303
|
-
export { ConceptoContextMenuComponent, ConceptoInputComponent, ConceptoMessageComponent, ConceptoTreeComponent, ConceptoUserControls, ConceptoUserControlsService, EntityComparisonComponent, EntityComparisonService };
|
|
2309
|
+
export { ConceptoContextMenuComponent, ConceptoInputComponent, ConceptoMessageComponent, ConceptoTreeComponent, ConceptoUserControls, ConceptoUserControlsService, DEFAULT_TREE_OPTIONS, EntityComparisonComponent, EntityComparisonService, TreeModel, TreeNode };
|
|
2304
2310
|
//# sourceMappingURL=concepto-user-controls.mjs.map
|