concepto-user-controls 0.0.8 → 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 (33) hide show
  1. package/esm2022/lib/concepto-tree/components/tree-node/tree-node.component.mjs +301 -0
  2. package/esm2022/lib/concepto-tree/components/tree-node-checkbox/tree-node-checkbox.component.mjs +90 -0
  3. package/esm2022/lib/concepto-tree/components/tree-node-content/tree-node-content.component.mjs +65 -0
  4. package/esm2022/lib/concepto-tree/components/tree-node-expander/tree-node-expander.component.mjs +74 -0
  5. package/esm2022/lib/concepto-tree/components/tree-root/tree-root.component.mjs +230 -0
  6. package/esm2022/lib/concepto-tree/components/tree-viewport/tree-viewport.component.mjs +216 -0
  7. package/esm2022/lib/concepto-tree/concepto-tree.component.mjs +214 -13
  8. package/esm2022/lib/concepto-tree/core/models/tree-events.model.mjs +2 -0
  9. package/esm2022/lib/concepto-tree/core/models/tree-node.model.mjs +102 -0
  10. package/esm2022/lib/concepto-tree/core/models/tree-options.model.mjs +24 -0
  11. package/esm2022/lib/concepto-tree/core/models/tree.model.mjs +313 -0
  12. package/esm2022/lib/concepto-tree/core/services/tree-drag-drop.service.mjs +27 -0
  13. package/esm2022/lib/concepto-tree/directives/tree-drag.directive.mjs +69 -0
  14. package/esm2022/lib/concepto-tree/directives/tree-drop.directive.mjs +124 -0
  15. package/esm2022/lib/concepto-tree/directives/tree-node-template.directive.mjs +19 -0
  16. package/fesm2022/concepto-user-controls.mjs +1818 -12
  17. package/fesm2022/concepto-user-controls.mjs.map +1 -1
  18. package/lib/concepto-tree/components/tree-node/tree-node.component.d.ts +19 -0
  19. package/lib/concepto-tree/components/tree-node-checkbox/tree-node-checkbox.component.d.ts +17 -0
  20. package/lib/concepto-tree/components/tree-node-content/tree-node-content.component.d.ts +18 -0
  21. package/lib/concepto-tree/components/tree-node-expander/tree-node-expander.component.d.ts +12 -0
  22. package/lib/concepto-tree/components/tree-root/tree-root.component.d.ts +35 -0
  23. package/lib/concepto-tree/components/tree-viewport/tree-viewport.component.d.ts +33 -0
  24. package/lib/concepto-tree/concepto-tree.component.d.ts +32 -1
  25. package/lib/concepto-tree/core/models/tree-events.model.d.ts +13 -0
  26. package/lib/concepto-tree/core/models/tree-node.model.d.ts +39 -0
  27. package/lib/concepto-tree/core/models/tree-options.model.d.ts +28 -0
  28. package/lib/concepto-tree/core/models/tree.model.d.ts +54 -0
  29. package/lib/concepto-tree/core/services/tree-drag-drop.service.d.ts +11 -0
  30. package/lib/concepto-tree/directives/tree-drag.directive.d.ts +16 -0
  31. package/lib/concepto-tree/directives/tree-drop.directive.d.ts +25 -0
  32. package/lib/concepto-tree/directives/tree-node-template.directive.d.ts +8 -0
  33. package/package.json +1 -1
@@ -0,0 +1,230 @@
1
+ // lib/components/tree-root/tree-root.component.ts
2
+ import { Component, Input, Output, EventEmitter, HostListener, signal, contentChild, TemplateRef, ChangeDetectionStrategy, } from '@angular/core';
3
+ import { CommonModule } from '@angular/common';
4
+ import { TreeModel } from '../../core/models/tree.model';
5
+ import { DEFAULT_TREE_OPTIONS } from '../../core/models/tree-options.model';
6
+ import { TreeNodeComponent } from '../tree-node/tree-node.component';
7
+ import { TreeViewportComponent } from '../tree-viewport/tree-viewport.component';
8
+ import { TreeNodeTemplateDirective } from '../../directives/tree-node-template.directive';
9
+ import * as i0 from "@angular/core";
10
+ export class TreeRootComponent {
11
+ nodes = [];
12
+ options = {};
13
+ initialized = new EventEmitter();
14
+ toggleExpanded = new EventEmitter();
15
+ activate = new EventEmitter();
16
+ deactivate = new EventEmitter();
17
+ select = new EventEmitter();
18
+ deselect = new EventEmitter();
19
+ focus = new EventEmitter();
20
+ blur = new EventEmitter();
21
+ moveNode = new EventEmitter();
22
+ loadChildren = new EventEmitter();
23
+ treeEvent = new EventEmitter();
24
+ // Content queries
25
+ nodeTemplate = contentChild(TreeNodeTemplateDirective, {
26
+ read: TemplateRef
27
+ });
28
+ treeModel;
29
+ mergedOptions;
30
+ isFocused = signal(false);
31
+ eventsSubscription;
32
+ ngOnInit() {
33
+ this.initializeTreeModel();
34
+ }
35
+ ngOnChanges(changes) {
36
+ if (changes['options'] && !changes['options'].firstChange) {
37
+ this.initializeTreeModel();
38
+ }
39
+ if (changes['nodes'] && !changes['nodes'].firstChange) {
40
+ this.treeModel.setData(this.nodes);
41
+ }
42
+ }
43
+ initializeTreeModel() {
44
+ // Unsubscribe from previous subscription if exists
45
+ this.eventsSubscription?.unsubscribe();
46
+ this.mergedOptions = {
47
+ ...DEFAULT_TREE_OPTIONS,
48
+ ...this.options,
49
+ };
50
+ this.treeModel = new TreeModel(this.mergedOptions);
51
+ this.treeModel.setData(this.nodes);
52
+ // Subscribe to tree events
53
+ this.eventsSubscription = this.treeModel.events$.subscribe(event => {
54
+ this.handleTreeEvent(event);
55
+ });
56
+ }
57
+ ngOnDestroy() {
58
+ // Clean up subscription
59
+ this.eventsSubscription?.unsubscribe();
60
+ }
61
+ handleTreeEvent(event) {
62
+ this.treeEvent.emit(event);
63
+ switch (event.type) {
64
+ case 'initialized':
65
+ this.initialized.emit(event);
66
+ break;
67
+ case 'expand':
68
+ case 'collapse':
69
+ this.toggleExpanded.emit(event);
70
+ break;
71
+ case 'activate':
72
+ this.activate.emit(event);
73
+ break;
74
+ case 'deactivate':
75
+ this.deactivate.emit(event);
76
+ break;
77
+ case 'select':
78
+ this.select.emit(event);
79
+ break;
80
+ case 'deselect':
81
+ this.deselect.emit(event);
82
+ break;
83
+ case 'focus':
84
+ this.focus.emit(event);
85
+ break;
86
+ case 'blur':
87
+ this.blur.emit(event);
88
+ break;
89
+ case 'moveNode':
90
+ this.moveNode.emit(event);
91
+ break;
92
+ case 'loadChildren':
93
+ this.loadChildren.emit(event);
94
+ break;
95
+ }
96
+ }
97
+ // Keyboard navigation
98
+ onKeydown(event) {
99
+ if (!this.isFocused())
100
+ return;
101
+ const key = event.key;
102
+ const isRtl = this.options.rtl;
103
+ switch (key) {
104
+ case 'ArrowDown':
105
+ this.treeModel.focusNextNode();
106
+ event.preventDefault();
107
+ break;
108
+ case 'ArrowUp':
109
+ this.treeModel.focusPreviousNode();
110
+ event.preventDefault();
111
+ break;
112
+ case 'ArrowRight':
113
+ isRtl ? this.treeModel.focusDrillUp() : this.treeModel.focusDrillDown();
114
+ event.preventDefault();
115
+ break;
116
+ case 'ArrowLeft':
117
+ isRtl ? this.treeModel.focusDrillDown() : this.treeModel.focusDrillUp();
118
+ event.preventDefault();
119
+ break;
120
+ case 'Enter':
121
+ case ' ':
122
+ const focused = this.treeModel.focusedNode();
123
+ if (focused) {
124
+ this.treeModel.activateNode(focused);
125
+ }
126
+ event.preventDefault();
127
+ break;
128
+ }
129
+ }
130
+ onFocus() {
131
+ this.isFocused.set(true);
132
+ }
133
+ onBlur() {
134
+ this.isFocused.set(false);
135
+ }
136
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeRootComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
137
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: TreeRootComponent, isStandalone: true, selector: "tree-root", inputs: { nodes: "nodes", options: "options" }, outputs: { initialized: "initialized", toggleExpanded: "toggleExpanded", activate: "activate", deactivate: "deactivate", select: "select", deselect: "deselect", focus: "focus", blur: "blur", moveNode: "moveNode", loadChildren: "loadChildren", treeEvent: "treeEvent" }, host: { listeners: { "keydown": "onKeydown($event)", "focus": "onFocus()", "blur": "onBlur()" }, properties: { "class.tree-focused": "isFocused()" } }, queries: [{ propertyName: "nodeTemplate", first: true, predicate: TreeNodeTemplateDirective, descendants: true, read: TemplateRef, isSignal: true }], usesOnChanges: true, ngImport: i0, template: `
138
+ <div
139
+ class="tree-container"
140
+ [class.tree-rtl]="mergedOptions?.rtl"
141
+ tabindex="0">
142
+
143
+ @if (mergedOptions?.useVirtualScroll) {
144
+ <tree-viewport
145
+ [treeModel]="treeModel"
146
+ [options]="mergedOptions"
147
+ [nodeTemplate]="nodeTemplate()">
148
+ </tree-viewport>
149
+ } @else {
150
+ @for (root of treeModel.roots(); track root.id) {
151
+ <tree-node
152
+ [node]="root"
153
+ [treeModel]="treeModel"
154
+ [options]="mergedOptions"
155
+ [nodeTemplate]="nodeTemplate()">
156
+ </tree-node>
157
+ }
158
+ }
159
+ </div>
160
+ `, isInline: true, styles: [".tree-container{width:100%;height:100%;outline:none}.tree-rtl{direction:rtl}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: TreeNodeComponent, selector: "tree-node", inputs: ["node", "treeModel", "options", "nodeTemplate"] }, { kind: "component", type: TreeViewportComponent, selector: "tree-viewport", inputs: ["treeModel", "options", "nodeTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
161
+ }
162
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeRootComponent, decorators: [{
163
+ type: Component,
164
+ args: [{ selector: 'tree-root', standalone: true, imports: [
165
+ CommonModule,
166
+ TreeNodeComponent,
167
+ TreeViewportComponent,
168
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: `
169
+ <div
170
+ class="tree-container"
171
+ [class.tree-rtl]="mergedOptions?.rtl"
172
+ tabindex="0">
173
+
174
+ @if (mergedOptions?.useVirtualScroll) {
175
+ <tree-viewport
176
+ [treeModel]="treeModel"
177
+ [options]="mergedOptions"
178
+ [nodeTemplate]="nodeTemplate()">
179
+ </tree-viewport>
180
+ } @else {
181
+ @for (root of treeModel.roots(); track root.id) {
182
+ <tree-node
183
+ [node]="root"
184
+ [treeModel]="treeModel"
185
+ [options]="mergedOptions"
186
+ [nodeTemplate]="nodeTemplate()">
187
+ </tree-node>
188
+ }
189
+ }
190
+ </div>
191
+ `, host: {
192
+ '[class.tree-focused]': 'isFocused()',
193
+ }, styles: [".tree-container{width:100%;height:100%;outline:none}.tree-rtl{direction:rtl}\n"] }]
194
+ }], propDecorators: { nodes: [{
195
+ type: Input
196
+ }], options: [{
197
+ type: Input
198
+ }], initialized: [{
199
+ type: Output
200
+ }], toggleExpanded: [{
201
+ type: Output
202
+ }], activate: [{
203
+ type: Output
204
+ }], deactivate: [{
205
+ type: Output
206
+ }], select: [{
207
+ type: Output
208
+ }], deselect: [{
209
+ type: Output
210
+ }], focus: [{
211
+ type: Output
212
+ }], blur: [{
213
+ type: Output
214
+ }], moveNode: [{
215
+ type: Output
216
+ }], loadChildren: [{
217
+ type: Output
218
+ }], treeEvent: [{
219
+ type: Output
220
+ }], onKeydown: [{
221
+ type: HostListener,
222
+ args: ['keydown', ['$event']]
223
+ }], onFocus: [{
224
+ type: HostListener,
225
+ args: ['focus']
226
+ }], onBlur: [{
227
+ type: HostListener,
228
+ args: ['blur']
229
+ }] } });
230
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree-root.component.js","sourceRoot":"","sources":["../../../../../../../projects/concepto-user-controls/src/lib/concepto-tree/components/tree-root/tree-root.component.ts"],"names":[],"mappings":"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","sourcesContent":["// lib/components/tree-root/tree-root.component.ts\r\nimport {\r\n  Component,\r\n  Input,\r\n  Output,\r\n  EventEmitter,\r\n  OnInit,\r\n  OnChanges,\r\n  OnDestroy,\r\n  SimpleChanges,\r\n  HostListener,\r\n  signal,\r\n  contentChild,\r\n  TemplateRef,\r\n  ChangeDetectionStrategy,\r\n} from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { Subscription } from 'rxjs';\r\nimport { TreeModel } from '../../core/models/tree.model';\r\nimport { TreeOptions, DEFAULT_TREE_OPTIONS } from '../../core/models/tree-options.model';\r\nimport { TreeEvent } from '../../core/models/tree-events.model';\r\nimport { TreeNodeComponent } from '../tree-node/tree-node.component';\r\nimport { TreeViewportComponent } from '../tree-viewport/tree-viewport.component';\r\nimport { TreeNodeTemplateDirective } from '../../directives/tree-node-template.directive';\r\n\r\n@Component({\r\n  selector: 'tree-root',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    TreeNodeComponent,\r\n    TreeViewportComponent,\r\n  ],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <div\r\n      class=\"tree-container\"\r\n      [class.tree-rtl]=\"mergedOptions?.rtl\"\r\n      tabindex=\"0\">\r\n\r\n      @if (mergedOptions?.useVirtualScroll) {\r\n        <tree-viewport\r\n          [treeModel]=\"treeModel\"\r\n          [options]=\"mergedOptions\"\r\n          [nodeTemplate]=\"nodeTemplate()\">\r\n        </tree-viewport>\r\n      } @else {\r\n        @for (root of treeModel.roots(); track root.id) {\r\n          <tree-node\r\n            [node]=\"root\"\r\n            [treeModel]=\"treeModel\"\r\n            [options]=\"mergedOptions\"\r\n            [nodeTemplate]=\"nodeTemplate()\">\r\n          </tree-node>\r\n        }\r\n      }\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    .tree-container {\r\n      width: 100%;\r\n      height: 100%;\r\n      outline: none;\r\n    }\r\n    \r\n    .tree-rtl {\r\n      direction: rtl;\r\n    }\r\n  `],\r\n  host: {\r\n    '[class.tree-focused]': 'isFocused()',\r\n  },\r\n})\r\nexport class TreeRootComponent implements OnInit, OnChanges, OnDestroy {\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  // Content queries\r\n  readonly nodeTemplate = contentChild(TreeNodeTemplateDirective, {\r\n    read: TemplateRef\r\n  });\r\n  \r\n  treeModel!: TreeModel;\r\n  mergedOptions!: TreeOptions;\r\n  isFocused = signal(false);\r\n\r\n  private eventsSubscription?: Subscription;\r\n  \r\n  ngOnInit(): void {\r\n    this.initializeTreeModel();\r\n  }\r\n  \r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['options'] && !changes['options'].firstChange) {\r\n      this.initializeTreeModel();\r\n    }\r\n    \r\n    if (changes['nodes'] && !changes['nodes'].firstChange) {\r\n      this.treeModel.setData(this.nodes);\r\n    }\r\n  }\r\n  \r\n  private initializeTreeModel(): void {\r\n    // Unsubscribe from previous subscription if exists\r\n    this.eventsSubscription?.unsubscribe();\r\n\r\n    this.mergedOptions = {\r\n      ...DEFAULT_TREE_OPTIONS,\r\n      ...this.options,\r\n    };\r\n\r\n    this.treeModel = new TreeModel(this.mergedOptions);\r\n    this.treeModel.setData(this.nodes);\r\n\r\n    // Subscribe to tree events\r\n    this.eventsSubscription = this.treeModel.events$.subscribe(event => {\r\n      this.handleTreeEvent(event);\r\n    });\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    // Clean up subscription\r\n    this.eventsSubscription?.unsubscribe();\r\n  }\r\n  \r\n  private handleTreeEvent(event: TreeEvent): void {\r\n    this.treeEvent.emit(event);\r\n    \r\n    switch (event.type) {\r\n      case 'initialized':\r\n        this.initialized.emit(event);\r\n        break;\r\n      case 'expand':\r\n      case 'collapse':\r\n        this.toggleExpanded.emit(event);\r\n        break;\r\n      case 'activate':\r\n        this.activate.emit(event);\r\n        break;\r\n      case 'deactivate':\r\n        this.deactivate.emit(event);\r\n        break;\r\n      case 'select':\r\n        this.select.emit(event);\r\n        break;\r\n      case 'deselect':\r\n        this.deselect.emit(event);\r\n        break;\r\n      case 'focus':\r\n        this.focus.emit(event);\r\n        break;\r\n      case 'blur':\r\n        this.blur.emit(event);\r\n        break;\r\n      case 'moveNode':\r\n        this.moveNode.emit(event);\r\n        break;\r\n      case 'loadChildren':\r\n        this.loadChildren.emit(event);\r\n        break;\r\n    }\r\n  }\r\n  \r\n  // Keyboard navigation\r\n  @HostListener('keydown', ['$event'])\r\n  onKeydown(event: KeyboardEvent): void {\r\n    if (!this.isFocused()) return;\r\n    \r\n    const key = event.key;\r\n    const isRtl = this.options.rtl;\r\n    \r\n    switch (key) {\r\n      case 'ArrowDown':\r\n        this.treeModel.focusNextNode();\r\n        event.preventDefault();\r\n        break;\r\n      case 'ArrowUp':\r\n        this.treeModel.focusPreviousNode();\r\n        event.preventDefault();\r\n        break;\r\n      case 'ArrowRight':\r\n        isRtl ? this.treeModel.focusDrillUp() : this.treeModel.focusDrillDown();\r\n        event.preventDefault();\r\n        break;\r\n      case 'ArrowLeft':\r\n        isRtl ? this.treeModel.focusDrillDown() : this.treeModel.focusDrillUp();\r\n        event.preventDefault();\r\n        break;\r\n      case 'Enter':\r\n      case ' ':\r\n        const focused = this.treeModel.focusedNode();\r\n        if (focused) {\r\n          this.treeModel.activateNode(focused);\r\n        }\r\n        event.preventDefault();\r\n        break;\r\n    }\r\n  }\r\n  \r\n  @HostListener('focus')\r\n  onFocus(): void {\r\n    this.isFocused.set(true);\r\n  }\r\n  \r\n  @HostListener('blur')\r\n  onBlur(): void {\r\n    this.isFocused.set(false);\r\n  }\r\n}"]}
@@ -0,0 +1,216 @@
1
+ // lib/components/tree-viewport/tree-viewport.component.ts
2
+ import { Component, Input, ViewChild, ChangeDetectionStrategy, signal, computed, effect, } from '@angular/core';
3
+ import { CommonModule } from '@angular/common';
4
+ import { TreeNodeComponent } from '../tree-node/tree-node.component';
5
+ import { fromEvent, Subject, takeUntil } from 'rxjs';
6
+ import { debounceTime } from 'rxjs/operators';
7
+ import * as i0 from "@angular/core";
8
+ export class TreeViewportComponent {
9
+ treeModel;
10
+ options;
11
+ nodeTemplate;
12
+ scrollContainer;
13
+ destroy$ = new Subject();
14
+ scrollTop = signal(0);
15
+ viewportHeight = signal(0);
16
+ flattenedNodes = computed(() => {
17
+ return this.treeModel.flattenedNodes();
18
+ });
19
+ totalHeight = computed(() => {
20
+ const nodes = this.flattenedNodes();
21
+ const nodeHeight = this.options.nodeHeight || 22;
22
+ if (typeof nodeHeight === 'number') {
23
+ return nodes.length * nodeHeight;
24
+ }
25
+ return nodes.reduce((sum, node) => sum + nodeHeight(node), 0);
26
+ });
27
+ visibleRange = computed(() => {
28
+ const scrollTop = this.scrollTop();
29
+ const viewportHeight = this.viewportHeight();
30
+ const nodes = this.flattenedNodes();
31
+ const nodeHeight = this.options.nodeHeight || 22;
32
+ const bufferAmount = this.options.bufferAmount || 5;
33
+ let startIndex = 0;
34
+ let accumulatedHeight = 0;
35
+ // Find start index
36
+ for (let i = 0; i < nodes.length; i++) {
37
+ const height = typeof nodeHeight === 'number'
38
+ ? nodeHeight
39
+ : nodeHeight(nodes[i]);
40
+ if (accumulatedHeight + height > scrollTop) {
41
+ startIndex = Math.max(0, i - bufferAmount);
42
+ break;
43
+ }
44
+ accumulatedHeight += height;
45
+ }
46
+ // Find end index
47
+ let endIndex = startIndex;
48
+ accumulatedHeight = 0;
49
+ for (let i = startIndex; i < nodes.length; i++) {
50
+ const height = typeof nodeHeight === 'number'
51
+ ? nodeHeight
52
+ : nodeHeight(nodes[i]);
53
+ accumulatedHeight += height;
54
+ if (accumulatedHeight > viewportHeight + (bufferAmount * (typeof nodeHeight === 'number' ? nodeHeight : 22))) {
55
+ endIndex = i;
56
+ break;
57
+ }
58
+ }
59
+ if (endIndex === startIndex) {
60
+ endIndex = nodes.length;
61
+ }
62
+ return { startIndex, endIndex };
63
+ });
64
+ visibleNodes = computed(() => {
65
+ const range = this.visibleRange();
66
+ const nodes = this.flattenedNodes();
67
+ return nodes.slice(range.startIndex, range.endIndex);
68
+ });
69
+ paddingTop = computed(() => {
70
+ const range = this.visibleRange();
71
+ const nodes = this.flattenedNodes();
72
+ const nodeHeight = this.options.nodeHeight || 22;
73
+ let height = 0;
74
+ for (let i = 0; i < range.startIndex; i++) {
75
+ height += typeof nodeHeight === 'number'
76
+ ? nodeHeight
77
+ : nodeHeight(nodes[i]);
78
+ }
79
+ return height;
80
+ });
81
+ paddingBottom = computed(() => {
82
+ const range = this.visibleRange();
83
+ const nodes = this.flattenedNodes();
84
+ const nodeHeight = this.options.nodeHeight || 22;
85
+ const totalHeight = this.totalHeight();
86
+ let visibleHeight = this.paddingTop();
87
+ for (let i = range.startIndex; i < range.endIndex; i++) {
88
+ visibleHeight += typeof nodeHeight === 'number'
89
+ ? nodeHeight
90
+ : nodeHeight(nodes[i]);
91
+ }
92
+ return Math.max(0, totalHeight - visibleHeight);
93
+ });
94
+ constructor() {
95
+ // Update viewport when tree changes
96
+ effect(() => {
97
+ this.flattenedNodes(); // Subscribe to changes
98
+ this.updateViewport();
99
+ });
100
+ }
101
+ ngOnInit() {
102
+ this.setupScrollListener();
103
+ this.updateViewportHeight();
104
+ }
105
+ ngOnDestroy() {
106
+ this.destroy$.next();
107
+ this.destroy$.complete();
108
+ }
109
+ setupScrollListener() {
110
+ fromEvent(this.scrollContainer.nativeElement, 'scroll')
111
+ .pipe(debounceTime(10), takeUntil(this.destroy$))
112
+ .subscribe(() => {
113
+ this.onScroll();
114
+ });
115
+ }
116
+ updateViewportHeight() {
117
+ const height = this.scrollContainer.nativeElement.clientHeight;
118
+ this.viewportHeight.set(height);
119
+ }
120
+ onScroll() {
121
+ const scrollTop = this.scrollContainer.nativeElement.scrollTop;
122
+ this.scrollTop.set(scrollTop);
123
+ }
124
+ updateViewport() {
125
+ // Force recalculation
126
+ this.scrollTop.set(this.scrollContainer.nativeElement.scrollTop);
127
+ }
128
+ scrollToNode(node) {
129
+ const nodes = this.flattenedNodes();
130
+ const index = nodes.findIndex(n => n.id === node.id);
131
+ if (index === -1)
132
+ return;
133
+ const nodeHeight = this.options.nodeHeight || 22;
134
+ let targetScrollTop = 0;
135
+ for (let i = 0; i < index; i++) {
136
+ targetScrollTop += typeof nodeHeight === 'number'
137
+ ? nodeHeight
138
+ : nodeHeight(nodes[i]);
139
+ }
140
+ this.scrollContainer.nativeElement.scrollTop = targetScrollTop;
141
+ }
142
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeViewportComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
143
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: TreeViewportComponent, isStandalone: true, selector: "tree-viewport", inputs: { treeModel: "treeModel", options: "options", nodeTemplate: "nodeTemplate" }, viewQueries: [{ propertyName: "scrollContainer", first: true, predicate: ["scrollContainer"], descendants: true, static: true }], ngImport: i0, template: `
144
+ <div
145
+ #scrollContainer
146
+ class="tree-viewport-container"
147
+ [style.height.px]="options.scrollContainerHeight || 400"
148
+ (scroll)="onScroll()">
149
+
150
+ <div class="tree-viewport-content"
151
+ [style.height.px]="totalHeight()">
152
+
153
+ <div class="tree-viewport-padding-top"
154
+ [style.height.px]="paddingTop()">
155
+ </div>
156
+
157
+ @for (node of visibleNodes(); track node.id) {
158
+ <tree-node
159
+ [node]="node"
160
+ [treeModel]="treeModel"
161
+ [options]="options"
162
+ [nodeTemplate]="nodeTemplate">
163
+ </tree-node>
164
+ }
165
+
166
+ <div class="tree-viewport-padding-bottom"
167
+ [style.height.px]="paddingBottom()">
168
+ </div>
169
+ </div>
170
+ </div>
171
+ `, isInline: true, styles: [".tree-viewport-container{overflow-y:auto;overflow-x:hidden;position:relative}.tree-viewport-content{position:relative;width:100%}.tree-viewport-padding-top,.tree-viewport-padding-bottom{width:100%}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: TreeNodeComponent, selector: "tree-node", inputs: ["node", "treeModel", "options", "nodeTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
172
+ }
173
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeViewportComponent, decorators: [{
174
+ type: Component,
175
+ args: [{ selector: 'tree-viewport', standalone: true, imports: [CommonModule, TreeNodeComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: `
176
+ <div
177
+ #scrollContainer
178
+ class="tree-viewport-container"
179
+ [style.height.px]="options.scrollContainerHeight || 400"
180
+ (scroll)="onScroll()">
181
+
182
+ <div class="tree-viewport-content"
183
+ [style.height.px]="totalHeight()">
184
+
185
+ <div class="tree-viewport-padding-top"
186
+ [style.height.px]="paddingTop()">
187
+ </div>
188
+
189
+ @for (node of visibleNodes(); track node.id) {
190
+ <tree-node
191
+ [node]="node"
192
+ [treeModel]="treeModel"
193
+ [options]="options"
194
+ [nodeTemplate]="nodeTemplate">
195
+ </tree-node>
196
+ }
197
+
198
+ <div class="tree-viewport-padding-bottom"
199
+ [style.height.px]="paddingBottom()">
200
+ </div>
201
+ </div>
202
+ </div>
203
+ `, styles: [".tree-viewport-container{overflow-y:auto;overflow-x:hidden;position:relative}.tree-viewport-content{position:relative;width:100%}.tree-viewport-padding-top,.tree-viewport-padding-bottom{width:100%}\n"] }]
204
+ }], ctorParameters: () => [], propDecorators: { treeModel: [{
205
+ type: Input,
206
+ args: [{ required: true }]
207
+ }], options: [{
208
+ type: Input,
209
+ args: [{ required: true }]
210
+ }], nodeTemplate: [{
211
+ type: Input
212
+ }], scrollContainer: [{
213
+ type: ViewChild,
214
+ args: ['scrollContainer', { static: true }]
215
+ }] } });
216
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree-viewport.component.js","sourceRoot":"","sources":["../../../../../../../projects/concepto-user-controls/src/lib/concepto-tree/components/tree-viewport/tree-viewport.component.ts"],"names":[],"mappings":"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","sourcesContent":["// lib/components/tree-viewport/tree-viewport.component.ts\r\nimport {\r\n  Component,\r\n  Input,\r\n  OnInit,\r\n  OnDestroy,\r\n  ViewChild,\r\n  ElementRef,\r\n  ChangeDetectionStrategy,\r\n  signal,\r\n  computed,\r\n  TemplateRef,\r\n  effect,\r\n} from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { TreeModel } from '../../core/models/tree.model';\r\nimport { TreeNode } from '../../core/models/tree-node.model';\r\nimport { TreeOptions } from '../../core/models/tree-options.model';\r\nimport { TreeNodeComponent } from '../tree-node/tree-node.component';\r\nimport { fromEvent, Subject, takeUntil } from 'rxjs';\r\nimport { debounceTime } from 'rxjs/operators';\r\n\r\n@Component({\r\n  selector: 'tree-viewport',\r\n  standalone: true,\r\n  imports: [CommonModule, TreeNodeComponent],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  template: `\r\n    <div \r\n      #scrollContainer\r\n      class=\"tree-viewport-container\"\r\n      [style.height.px]=\"options.scrollContainerHeight || 400\"\r\n      (scroll)=\"onScroll()\">\r\n      \r\n      <div class=\"tree-viewport-content\"\r\n           [style.height.px]=\"totalHeight()\">\r\n        \r\n        <div class=\"tree-viewport-padding-top\"\r\n             [style.height.px]=\"paddingTop()\">\r\n        </div>\r\n        \r\n        @for (node of visibleNodes(); track node.id) {\r\n          <tree-node\r\n            [node]=\"node\"\r\n            [treeModel]=\"treeModel\"\r\n            [options]=\"options\"\r\n            [nodeTemplate]=\"nodeTemplate\">\r\n          </tree-node>\r\n        }\r\n        \r\n        <div class=\"tree-viewport-padding-bottom\"\r\n             [style.height.px]=\"paddingBottom()\">\r\n        </div>\r\n      </div>\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    .tree-viewport-container {\r\n      overflow-y: auto;\r\n      overflow-x: hidden;\r\n      position: relative;\r\n    }\r\n    \r\n    .tree-viewport-content {\r\n      position: relative;\r\n      width: 100%;\r\n    }\r\n    \r\n    .tree-viewport-padding-top,\r\n    .tree-viewport-padding-bottom {\r\n      width: 100%;\r\n    }\r\n  `],\r\n})\r\nexport class TreeViewportComponent implements OnInit, OnDestroy {\r\n  @Input({ required: true }) treeModel!: TreeModel;\r\n  @Input({ required: true }) options!: TreeOptions;\r\n  @Input() nodeTemplate?: TemplateRef<any>;\r\n  \r\n  @ViewChild('scrollContainer', { static: true }) \r\n  scrollContainer!: ElementRef<HTMLElement>;\r\n  \r\n  private destroy$ = new Subject<void>();\r\n  \r\n  readonly scrollTop = signal(0);\r\n  readonly viewportHeight = signal(0);\r\n  \r\n  readonly flattenedNodes = computed(() => {\r\n    return this.treeModel.flattenedNodes();\r\n  });\r\n  \r\n  readonly totalHeight = computed(() => {\r\n    const nodes = this.flattenedNodes();\r\n    const nodeHeight = this.options.nodeHeight || 22;\r\n    \r\n    if (typeof nodeHeight === 'number') {\r\n      return nodes.length * nodeHeight;\r\n    }\r\n    \r\n    return nodes.reduce((sum, node) => sum + nodeHeight(node), 0);\r\n  });\r\n  \r\n  readonly visibleRange = computed(() => {\r\n    const scrollTop = this.scrollTop();\r\n    const viewportHeight = this.viewportHeight();\r\n    const nodes = this.flattenedNodes();\r\n    const nodeHeight = this.options.nodeHeight || 22;\r\n    const bufferAmount = this.options.bufferAmount || 5;\r\n    \r\n    let startIndex = 0;\r\n    let accumulatedHeight = 0;\r\n    \r\n    // Find start index\r\n    for (let i = 0; i < nodes.length; i++) {\r\n      const height = typeof nodeHeight === 'number' \r\n        ? nodeHeight \r\n        : nodeHeight(nodes[i]);\r\n      \r\n      if (accumulatedHeight + height > scrollTop) {\r\n        startIndex = Math.max(0, i - bufferAmount);\r\n        break;\r\n      }\r\n      accumulatedHeight += height;\r\n    }\r\n    \r\n    // Find end index\r\n    let endIndex = startIndex;\r\n    accumulatedHeight = 0;\r\n    \r\n    for (let i = startIndex; i < nodes.length; i++) {\r\n      const height = typeof nodeHeight === 'number'\r\n        ? nodeHeight\r\n        : nodeHeight(nodes[i]);\r\n      \r\n      accumulatedHeight += height;\r\n      \r\n      if (accumulatedHeight > viewportHeight + (bufferAmount * (typeof nodeHeight === 'number' ? nodeHeight : 22))) {\r\n        endIndex = i;\r\n        break;\r\n      }\r\n    }\r\n    \r\n    if (endIndex === startIndex) {\r\n      endIndex = nodes.length;\r\n    }\r\n    \r\n    return { startIndex, endIndex };\r\n  });\r\n  \r\n  readonly visibleNodes = computed(() => {\r\n    const range = this.visibleRange();\r\n    const nodes = this.flattenedNodes();\r\n    return nodes.slice(range.startIndex, range.endIndex);\r\n  });\r\n  \r\n  readonly paddingTop = computed(() => {\r\n    const range = this.visibleRange();\r\n    const nodes = this.flattenedNodes();\r\n    const nodeHeight = this.options.nodeHeight || 22;\r\n    \r\n    let height = 0;\r\n    for (let i = 0; i < range.startIndex; i++) {\r\n      height += typeof nodeHeight === 'number' \r\n        ? nodeHeight \r\n        : nodeHeight(nodes[i]);\r\n    }\r\n    \r\n    return height;\r\n  });\r\n  \r\n  readonly paddingBottom = computed(() => {\r\n    const range = this.visibleRange();\r\n    const nodes = this.flattenedNodes();\r\n    const nodeHeight = this.options.nodeHeight || 22;\r\n    const totalHeight = this.totalHeight();\r\n    \r\n    let visibleHeight = this.paddingTop();\r\n    \r\n    for (let i = range.startIndex; i < range.endIndex; i++) {\r\n      visibleHeight += typeof nodeHeight === 'number'\r\n        ? nodeHeight\r\n        : nodeHeight(nodes[i]);\r\n    }\r\n    \r\n    return Math.max(0, totalHeight - visibleHeight);\r\n  });\r\n  \r\n  constructor() {\r\n    // Update viewport when tree changes\r\n    effect(() => {\r\n      this.flattenedNodes(); // Subscribe to changes\r\n      this.updateViewport();\r\n    });\r\n  }\r\n  \r\n  ngOnInit(): void {\r\n    this.setupScrollListener();\r\n    this.updateViewportHeight();\r\n  }\r\n  \r\n  ngOnDestroy(): void {\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n  }\r\n  \r\n  private setupScrollListener(): void {\r\n    fromEvent(this.scrollContainer.nativeElement, 'scroll')\r\n      .pipe(\r\n        debounceTime(10),\r\n        takeUntil(this.destroy$)\r\n      )\r\n      .subscribe(() => {\r\n        this.onScroll();\r\n      });\r\n  }\r\n  \r\n  private updateViewportHeight(): void {\r\n    const height = this.scrollContainer.nativeElement.clientHeight;\r\n    this.viewportHeight.set(height);\r\n  }\r\n  \r\n  onScroll(): void {\r\n    const scrollTop = this.scrollContainer.nativeElement.scrollTop;\r\n    this.scrollTop.set(scrollTop);\r\n  }\r\n  \r\n  private updateViewport(): void {\r\n    // Force recalculation\r\n    this.scrollTop.set(this.scrollContainer.nativeElement.scrollTop);\r\n  }\r\n  \r\n  scrollToNode(node: TreeNode): void {\r\n    const nodes = this.flattenedNodes();\r\n    const index = nodes.findIndex(n => n.id === node.id);\r\n    \r\n    if (index === -1) return;\r\n    \r\n    const nodeHeight = this.options.nodeHeight || 22;\r\n    let targetScrollTop = 0;\r\n    \r\n    for (let i = 0; i < index; i++) {\r\n      targetScrollTop += typeof nodeHeight === 'number'\r\n        ? nodeHeight\r\n        : nodeHeight(nodes[i]);\r\n    }\r\n    \r\n    this.scrollContainer.nativeElement.scrollTop = targetScrollTop;\r\n  }\r\n}"]}