concepto-user-controls 0.0.7 → 0.0.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (41) hide show
  1. package/esm2022/lib/concepto-context-menu/concepto-context-menu.component.mjs +3 -7
  2. package/esm2022/lib/concepto-tree/components/tree-node/tree-node.component.mjs +301 -0
  3. package/esm2022/lib/concepto-tree/components/tree-node-checkbox/tree-node-checkbox.component.mjs +90 -0
  4. package/esm2022/lib/concepto-tree/components/tree-node-content/tree-node-content.component.mjs +65 -0
  5. package/esm2022/lib/concepto-tree/components/tree-node-expander/tree-node-expander.component.mjs +74 -0
  6. package/esm2022/lib/concepto-tree/components/tree-root/tree-root.component.mjs +230 -0
  7. package/esm2022/lib/concepto-tree/components/tree-viewport/tree-viewport.component.mjs +216 -0
  8. package/esm2022/lib/concepto-tree/concepto-tree.component.mjs +220 -0
  9. package/esm2022/lib/concepto-tree/core/models/tree-events.model.mjs +2 -0
  10. package/esm2022/lib/concepto-tree/core/models/tree-node.model.mjs +102 -0
  11. package/esm2022/lib/concepto-tree/core/models/tree-options.model.mjs +24 -0
  12. package/esm2022/lib/concepto-tree/core/models/tree.model.mjs +313 -0
  13. package/esm2022/lib/concepto-tree/core/services/tree-drag-drop.service.mjs +27 -0
  14. package/esm2022/lib/concepto-tree/directives/tree-drag.directive.mjs +69 -0
  15. package/esm2022/lib/concepto-tree/directives/tree-drop.directive.mjs +124 -0
  16. package/esm2022/lib/concepto-tree/directives/tree-node-template.directive.mjs +19 -0
  17. package/esm2022/lib/entity-comparison/components/entity-comparison.component.mjs +218 -0
  18. package/esm2022/lib/entity-comparison/core/services/entity-comparison.service.mjs +111 -0
  19. package/esm2022/public-api.mjs +4 -1
  20. package/fesm2022/concepto-user-controls.mjs +2147 -7
  21. package/fesm2022/concepto-user-controls.mjs.map +1 -1
  22. package/lib/concepto-context-menu/concepto-context-menu.component.d.ts +2 -3
  23. package/lib/concepto-tree/components/tree-node/tree-node.component.d.ts +19 -0
  24. package/lib/concepto-tree/components/tree-node-checkbox/tree-node-checkbox.component.d.ts +17 -0
  25. package/lib/concepto-tree/components/tree-node-content/tree-node-content.component.d.ts +18 -0
  26. package/lib/concepto-tree/components/tree-node-expander/tree-node-expander.component.d.ts +12 -0
  27. package/lib/concepto-tree/components/tree-root/tree-root.component.d.ts +35 -0
  28. package/lib/concepto-tree/components/tree-viewport/tree-viewport.component.d.ts +33 -0
  29. package/lib/concepto-tree/concepto-tree.component.d.ts +36 -0
  30. package/lib/concepto-tree/core/models/tree-events.model.d.ts +13 -0
  31. package/lib/concepto-tree/core/models/tree-node.model.d.ts +39 -0
  32. package/lib/concepto-tree/core/models/tree-options.model.d.ts +28 -0
  33. package/lib/concepto-tree/core/models/tree.model.d.ts +54 -0
  34. package/lib/concepto-tree/core/services/tree-drag-drop.service.d.ts +11 -0
  35. package/lib/concepto-tree/directives/tree-drag.directive.d.ts +16 -0
  36. package/lib/concepto-tree/directives/tree-drop.directive.d.ts +25 -0
  37. package/lib/concepto-tree/directives/tree-node-template.directive.d.ts +8 -0
  38. package/lib/entity-comparison/components/entity-comparison.component.d.ts +49 -0
  39. package/lib/entity-comparison/core/services/entity-comparison.service.d.ts +10 -0
  40. package/package.json +1 -1
  41. package/public-api.d.ts +3 -0
@@ -1,11 +1,10 @@
1
- import { Component, EventEmitter, HostListener, Input, Output } from '@angular/core';
1
+ import { Component, HostListener, Input } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
3
  import * as i0 from "@angular/core";
4
4
  import * as i1 from "@angular/common";
5
5
  export class ConceptoContextMenuComponent {
6
6
  elRef;
7
7
  nodes = [];
8
- itemSelected = new EventEmitter();
9
8
  menuTree = [];
10
9
  visible = false;
11
10
  pos = { x: 0, y: 0 };
@@ -49,7 +48,6 @@ export class ConceptoContextMenuComponent {
49
48
  onOptionClick(option) {
50
49
  if (!option.children || option.children.length === 0) {
51
50
  console.log('Selected:', option);
52
- this.itemSelected.emit(option.NodesId);
53
51
  this.hide();
54
52
  }
55
53
  }
@@ -57,17 +55,15 @@ export class ConceptoContextMenuComponent {
57
55
  return !!option.children && option.children.length > 0;
58
56
  }
59
57
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ConceptoContextMenuComponent, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
60
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ConceptoContextMenuComponent, isStandalone: true, selector: "concepto-context-menu", inputs: { nodes: "nodes" }, outputs: { itemSelected: "itemSelected" }, host: { listeners: { "document:click": "onOutsideClick($event)" } }, usesOnChanges: true, ngImport: i0, template: "<div class=\"context-menu\" *ngIf=\"visible\" [ngStyle]=\"{ top: pos.y + 'px', left: pos.x + 'px' }\">\r\n <ul class=\"menu-root\">\r\n <ng-container *ngFor=\"let item of menuTree\">\r\n <ng-container *ngTemplateOutlet=\"renderNode; context: { $implicit: item }\"></ng-container>\r\n </ng-container>\r\n </ul>\r\n</div>\r\n\r\n<ng-template #renderNode let-node>\r\n <li class=\"menu-item\" [class.has-children]=\"hasChildren(node)\">\r\n <div (click)=\"onOptionClick(node)\">\r\n @if (node.NodesImg) {\r\n <img [src]=\"node.NodesImg\" class=\"icon\" />\r\n }\r\n @if (!node.NodesImg) {\r\n <div class=\"icon\"></div>\r\n }\r\n {{ node.NodesText }}\r\n </div>\r\n <ul class=\"submenu\" *ngIf=\"hasChildren(node)\">\r\n <ng-container *ngFor=\"let child of node.children\">\r\n <ng-container *ngTemplateOutlet=\"renderNode; context: { $implicit: child }\"></ng-container>\r\n </ng-container>\r\n </ul>\r\n </li>\r\n</ng-template>", styles: [".context-menu{position:fixed;z-index:9999;background-color:#fff;border:1px solid #ccc;box-shadow:2px 2px 8px #00000026;min-width:200px;font-family:sans-serif}.menu-root,.submenu{list-style:none;margin:0;padding:0}.menu-item{position:relative}.menu-item>div{padding:8px 12px;cursor:pointer;display:flex;align-items:center;white-space:nowrap}.menu-item>div:hover{background-color:#f0f0f0}.menu-item.has-children>div:after{content:\"\\25b6\";margin-left:auto;font-size:10px}.submenu{display:none;position:absolute;top:0;left:100%;border:1px solid #ccc;background-color:#fff;min-width:180px;box-shadow:2px 2px 6px #0000001a}.menu-item:hover>.submenu{display:block}.icon{width:16px;height:16px;margin-right:8px}\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: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
58
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ConceptoContextMenuComponent, isStandalone: true, selector: "concepto-context-menu", inputs: { nodes: "nodes" }, host: { listeners: { "document:click": "onOutsideClick($event)" } }, usesOnChanges: true, ngImport: i0, template: "<div class=\"context-menu\" *ngIf=\"visible\" [ngStyle]=\"{ top: pos.y + 'px', left: pos.x + 'px' }\">\r\n <ul class=\"menu-root\">\r\n <ng-container *ngFor=\"let item of menuTree\">\r\n <ng-container *ngTemplateOutlet=\"renderNode; context: { $implicit: item }\"></ng-container>\r\n </ng-container>\r\n </ul>\r\n</div>\r\n\r\n<ng-template #renderNode let-node>\r\n <li class=\"menu-item\" [class.has-children]=\"hasChildren(node)\">\r\n <div (click)=\"onOptionClick(node)\">\r\n @if (node.NodesImg) {\r\n <img [src]=\"node.NodesImg\" class=\"icon\" />\r\n }\r\n @if (!node.NodesImg) {\r\n <div class=\"icon\"></div>\r\n }\r\n {{ node.NodesText }}\r\n </div>\r\n <ul class=\"submenu\" *ngIf=\"hasChildren(node)\">\r\n <ng-container *ngFor=\"let child of node.children\">\r\n <ng-container *ngTemplateOutlet=\"renderNode; context: { $implicit: child }\"></ng-container>\r\n </ng-container>\r\n </ul>\r\n </li>\r\n</ng-template>", styles: [".context-menu{position:fixed;z-index:9999;background-color:#fff;border:1px solid #ccc;box-shadow:2px 2px 8px #00000026;min-width:200px;font-family:sans-serif}.menu-root,.submenu{list-style:none;margin:0;padding:0}.menu-item{position:relative}.menu-item>div{padding:8px 12px;cursor:pointer;display:flex;align-items:center;white-space:nowrap}.menu-item>div:hover{background-color:#f0f0f0}.menu-item.has-children>div:after{content:\"\\25b6\";margin-left:auto;font-size:10px}.submenu{display:none;position:absolute;top:0;left:100%;border:1px solid #ccc;background-color:#fff;min-width:180px;box-shadow:2px 2px 6px #0000001a}.menu-item:hover>.submenu{display:block}.icon{width:16px;height:16px;margin-right:8px}\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: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
61
59
  }
62
60
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ConceptoContextMenuComponent, decorators: [{
63
61
  type: Component,
64
62
  args: [{ selector: 'concepto-context-menu', standalone: true, imports: [CommonModule], template: "<div class=\"context-menu\" *ngIf=\"visible\" [ngStyle]=\"{ top: pos.y + 'px', left: pos.x + 'px' }\">\r\n <ul class=\"menu-root\">\r\n <ng-container *ngFor=\"let item of menuTree\">\r\n <ng-container *ngTemplateOutlet=\"renderNode; context: { $implicit: item }\"></ng-container>\r\n </ng-container>\r\n </ul>\r\n</div>\r\n\r\n<ng-template #renderNode let-node>\r\n <li class=\"menu-item\" [class.has-children]=\"hasChildren(node)\">\r\n <div (click)=\"onOptionClick(node)\">\r\n @if (node.NodesImg) {\r\n <img [src]=\"node.NodesImg\" class=\"icon\" />\r\n }\r\n @if (!node.NodesImg) {\r\n <div class=\"icon\"></div>\r\n }\r\n {{ node.NodesText }}\r\n </div>\r\n <ul class=\"submenu\" *ngIf=\"hasChildren(node)\">\r\n <ng-container *ngFor=\"let child of node.children\">\r\n <ng-container *ngTemplateOutlet=\"renderNode; context: { $implicit: child }\"></ng-container>\r\n </ng-container>\r\n </ul>\r\n </li>\r\n</ng-template>", styles: [".context-menu{position:fixed;z-index:9999;background-color:#fff;border:1px solid #ccc;box-shadow:2px 2px 8px #00000026;min-width:200px;font-family:sans-serif}.menu-root,.submenu{list-style:none;margin:0;padding:0}.menu-item{position:relative}.menu-item>div{padding:8px 12px;cursor:pointer;display:flex;align-items:center;white-space:nowrap}.menu-item>div:hover{background-color:#f0f0f0}.menu-item.has-children>div:after{content:\"\\25b6\";margin-left:auto;font-size:10px}.submenu{display:none;position:absolute;top:0;left:100%;border:1px solid #ccc;background-color:#fff;min-width:180px;box-shadow:2px 2px 6px #0000001a}.menu-item:hover>.submenu{display:block}.icon{width:16px;height:16px;margin-right:8px}\n"] }]
65
63
  }], ctorParameters: () => [{ type: i0.ElementRef }], propDecorators: { nodes: [{
66
64
  type: Input
67
- }], itemSelected: [{
68
- type: Output
69
65
  }], onOutsideClick: [{
70
66
  type: HostListener,
71
67
  args: ['document:click', ['$event']]
72
68
  }] } });
73
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"concepto-context-menu.component.js","sourceRoot":"","sources":["../../../../../projects/concepto-user-controls/src/lib/concepto-context-menu/concepto-context-menu.component.ts","../../../../../projects/concepto-user-controls/src/lib/concepto-context-menu/concepto-context-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, HostListener, Input, Output } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\n\r\n\r\nexport interface NodeOption {\r\n  NodesId: string;\r\n  NodesText: string;\r\n  NodesParentId: string;\r\n  NodesImg: string;\r\n  children?: NodeOption[];\r\n}\r\n\r\n@Component({\r\n  selector: 'concepto-context-menu',\r\n  standalone: true,\r\n  imports: [CommonModule],\r\n  templateUrl: './concepto-context-menu.component.html',\r\n  styleUrl: './concepto-context-menu.component.css'\r\n})\r\nexport class ConceptoContextMenuComponent {\r\n\r\n  @Input() nodes: NodeOption[] = [];\r\n  @Output() itemSelected = new EventEmitter<string>();\r\n  menuTree: NodeOption[] = [];\r\n  visible = false;\r\n  pos = { x: 0, y: 0 };\r\n\r\n  constructor(private elRef: ElementRef) {}\r\n\r\n  ngOnChanges(): void {\r\n    const map = new Map<string, NodeOption>();\r\n    this.nodes.forEach(n => map.set(n.NodesId, { ...n, children: [] }));\r\n\r\n    const roots: NodeOption[] = [];\r\n    map.forEach(n => {\r\n      if (n.NodesParentId && map.has(n.NodesParentId)) {\r\n        map.get(n.NodesParentId)?.children?.push(n);\r\n      } else {\r\n        roots.push(n);\r\n      }\r\n    });\r\n    this.menuTree = roots;\r\n  }\r\n\r\n  showAtEvent(event: MouseEvent): void {\r\n    event.preventDefault();\r\n    const x = event.clientX;\r\n    const y = event.clientY;\r\n    this.visible = true;\r\n    setTimeout(() => {\r\n      const menu = this.elRef.nativeElement.querySelector('.context-menu');\r\n      const rect = menu.getBoundingClientRect();\r\n      this.pos.x = x + rect.width > window.innerWidth ? window.innerWidth - rect.width - 10 : x;\r\n      this.pos.y = y + rect.height > window.innerHeight ? window.innerHeight - rect.height - 10 : y;\r\n    });\r\n  }\r\n\r\n  hide(): void {\r\n    this.visible = false;\r\n  }\r\n\r\n  @HostListener('document:click', ['$event'])\r\n  onOutsideClick(event: MouseEvent): void {\r\n    if (!this.elRef.nativeElement.contains(event.target)) {\r\n      this.hide();\r\n    }\r\n  }\r\n\r\n  onOptionClick(option: NodeOption): void {\r\n    if (!option.children || option.children.length === 0) {\r\n      console.log('Selected:', option);\r\n      this.itemSelected.emit(option.NodesId);\r\n      this.hide();\r\n    }\r\n  }\r\n\r\n  hasChildren(option: NodeOption): boolean {\r\n    return !!option.children && option.children.length > 0;\r\n  }\r\n}\r\n\r\n","<div class=\"context-menu\" *ngIf=\"visible\" [ngStyle]=\"{ top: pos.y + 'px', left: pos.x + 'px' }\">\r\n    <ul class=\"menu-root\">\r\n        <ng-container *ngFor=\"let item of menuTree\">\r\n            <ng-container *ngTemplateOutlet=\"renderNode; context: { $implicit: item }\"></ng-container>\r\n        </ng-container>\r\n    </ul>\r\n</div>\r\n\r\n<ng-template #renderNode let-node>\r\n    <li class=\"menu-item\" [class.has-children]=\"hasChildren(node)\">\r\n        <div (click)=\"onOptionClick(node)\">\r\n            @if (node.NodesImg) {\r\n            <img [src]=\"node.NodesImg\" class=\"icon\" />\r\n            }\r\n            @if (!node.NodesImg) {\r\n            <div class=\"icon\"></div>\r\n            }\r\n            {{ node.NodesText }}\r\n        </div>\r\n        <ul class=\"submenu\" *ngIf=\"hasChildren(node)\">\r\n            <ng-container *ngFor=\"let child of node.children\">\r\n                <ng-container *ngTemplateOutlet=\"renderNode; context: { $implicit: child }\"></ng-container>\r\n            </ng-container>\r\n        </ul>\r\n    </li>\r\n</ng-template>"]}
69
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"concepto-context-menu.component.js","sourceRoot":"","sources":["../../../../../projects/concepto-user-controls/src/lib/concepto-context-menu/concepto-context-menu.component.ts","../../../../../projects/concepto-user-controls/src/lib/concepto-context-menu/concepto-context-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, HostListener, Input } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\n\r\n\r\nexport interface NodeOption {\r\n  NodesId: string;\r\n  NodesText: string;\r\n  NodesParentId: string;\r\n  NodesImg: string;\r\n  children?: NodeOption[];\r\n}\r\n\r\n@Component({\r\n  selector: 'concepto-context-menu',\r\n  standalone: true,\r\n  imports: [CommonModule],\r\n  templateUrl: './concepto-context-menu.component.html',\r\n  styleUrl: './concepto-context-menu.component.css'\r\n})\r\nexport class ConceptoContextMenuComponent {\r\n\r\n  @Input() nodes: NodeOption[] = [];\r\n  menuTree: NodeOption[] = [];\r\n  visible = false;\r\n  pos = { x: 0, y: 0 };\r\n\r\n  constructor(private elRef: ElementRef) {}\r\n\r\n  ngOnChanges(): void {\r\n    const map = new Map<string, NodeOption>();\r\n    this.nodes.forEach(n => map.set(n.NodesId, { ...n, children: [] }));\r\n\r\n    const roots: NodeOption[] = [];\r\n    map.forEach(n => {\r\n      if (n.NodesParentId && map.has(n.NodesParentId)) {\r\n        map.get(n.NodesParentId)?.children?.push(n);\r\n      } else {\r\n        roots.push(n);\r\n      }\r\n    });\r\n    this.menuTree = roots;\r\n  }\r\n\r\n  showAtEvent(event: MouseEvent): void {\r\n    event.preventDefault();\r\n    const x = event.clientX;\r\n    const y = event.clientY;\r\n    this.visible = true;\r\n    setTimeout(() => {\r\n      const menu = this.elRef.nativeElement.querySelector('.context-menu');\r\n      const rect = menu.getBoundingClientRect();\r\n      this.pos.x = x + rect.width > window.innerWidth ? window.innerWidth - rect.width - 10 : x;\r\n      this.pos.y = y + rect.height > window.innerHeight ? window.innerHeight - rect.height - 10 : y;\r\n    });\r\n  }\r\n\r\n  hide(): void {\r\n    this.visible = false;\r\n  }\r\n\r\n  @HostListener('document:click', ['$event'])\r\n  onOutsideClick(event: MouseEvent): void {\r\n    if (!this.elRef.nativeElement.contains(event.target)) {\r\n      this.hide();\r\n    }\r\n  }\r\n\r\n  onOptionClick(option: NodeOption): void {\r\n    if (!option.children || option.children.length === 0) {\r\n      console.log('Selected:', option);\r\n      this.hide();\r\n    }\r\n  }\r\n\r\n  hasChildren(option: NodeOption): boolean {\r\n    return !!option.children && option.children.length > 0;\r\n  }\r\n}\r\n\r\n","<div class=\"context-menu\" *ngIf=\"visible\" [ngStyle]=\"{ top: pos.y + 'px', left: pos.x + 'px' }\">\r\n    <ul class=\"menu-root\">\r\n        <ng-container *ngFor=\"let item of menuTree\">\r\n            <ng-container *ngTemplateOutlet=\"renderNode; context: { $implicit: item }\"></ng-container>\r\n        </ng-container>\r\n    </ul>\r\n</div>\r\n\r\n<ng-template #renderNode let-node>\r\n    <li class=\"menu-item\" [class.has-children]=\"hasChildren(node)\">\r\n        <div (click)=\"onOptionClick(node)\">\r\n            @if (node.NodesImg) {\r\n            <img [src]=\"node.NodesImg\" class=\"icon\" />\r\n            }\r\n            @if (!node.NodesImg) {\r\n            <div class=\"icon\"></div>\r\n            }\r\n            {{ node.NodesText }}\r\n        </div>\r\n        <ul class=\"submenu\" *ngIf=\"hasChildren(node)\">\r\n            <ng-container *ngFor=\"let child of node.children\">\r\n                <ng-container *ngTemplateOutlet=\"renderNode; context: { $implicit: child }\"></ng-container>\r\n            </ng-container>\r\n        </ul>\r\n    </li>\r\n</ng-template>"]}
@@ -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}"]}
@@ -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
@@ -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=