concepto-user-controls 0.0.8 → 0.0.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (37) hide show
  1. package/esm2022/lib/concepto-input/concepto-input.component.mjs +20 -0
  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 +214 -13
  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/public-api.mjs +2 -1
  18. package/fesm2022/concepto-user-controls.mjs +1837 -13
  19. package/fesm2022/concepto-user-controls.mjs.map +1 -1
  20. package/lib/concepto-input/concepto-input.component.d.ts +8 -0
  21. package/lib/concepto-tree/components/tree-node/tree-node.component.d.ts +19 -0
  22. package/lib/concepto-tree/components/tree-node-checkbox/tree-node-checkbox.component.d.ts +17 -0
  23. package/lib/concepto-tree/components/tree-node-content/tree-node-content.component.d.ts +18 -0
  24. package/lib/concepto-tree/components/tree-node-expander/tree-node-expander.component.d.ts +12 -0
  25. package/lib/concepto-tree/components/tree-root/tree-root.component.d.ts +35 -0
  26. package/lib/concepto-tree/components/tree-viewport/tree-viewport.component.d.ts +33 -0
  27. package/lib/concepto-tree/concepto-tree.component.d.ts +32 -1
  28. package/lib/concepto-tree/core/models/tree-events.model.d.ts +13 -0
  29. package/lib/concepto-tree/core/models/tree-node.model.d.ts +39 -0
  30. package/lib/concepto-tree/core/models/tree-options.model.d.ts +28 -0
  31. package/lib/concepto-tree/core/models/tree.model.d.ts +54 -0
  32. package/lib/concepto-tree/core/services/tree-drag-drop.service.d.ts +11 -0
  33. package/lib/concepto-tree/directives/tree-drag.directive.d.ts +16 -0
  34. package/lib/concepto-tree/directives/tree-drop.directive.d.ts +25 -0
  35. package/lib/concepto-tree/directives/tree-node-template.directive.d.ts +8 -0
  36. package/package.json +1 -1
  37. package/public-api.d.ts +1 -0
@@ -0,0 +1,313 @@
1
+ // lib/core/models/tree.model.ts
2
+ import { signal, computed } from '@angular/core';
3
+ import { Subject } from 'rxjs';
4
+ import { TreeNode } from './tree-node.model';
5
+ export class TreeModel {
6
+ options;
7
+ // Reactive state with signals
8
+ roots = signal([]);
9
+ focusedNodeId = signal(null);
10
+ expandedNodeIds = signal(new Set());
11
+ activeNodeIds = signal(new Set());
12
+ selectedNodeIds = signal(new Set());
13
+ hiddenNodeIds = signal(new Set());
14
+ // Computed signals
15
+ focusedNode = computed(() => {
16
+ const id = this.focusedNodeId();
17
+ return id ? this.getNodeById(id) : null;
18
+ });
19
+ expandedNodes = computed(() => this.getAllNodes().filter(node => this.expandedNodeIds().has(node.id)));
20
+ activeNodes = computed(() => this.getAllNodes().filter(node => this.activeNodeIds().has(node.id)));
21
+ selectedNodes = computed(() => this.getAllNodes().filter(node => this.selectedNodeIds().has(node.id)));
22
+ visibleNodes = computed(() => this.getAllNodes().filter(node => !node.isHidden()));
23
+ flattenedNodes = computed(() => {
24
+ const flattened = [];
25
+ const traverse = (nodes) => {
26
+ nodes.forEach(node => {
27
+ if (!node.isHidden()) {
28
+ flattened.push(node);
29
+ if (node.isExpanded() && node.children.length > 0) {
30
+ traverse(node.children);
31
+ }
32
+ }
33
+ });
34
+ };
35
+ traverse(this.roots());
36
+ return flattened;
37
+ });
38
+ // Event streams
39
+ events$ = new Subject();
40
+ // Virtual root for unified handling
41
+ virtualRoot = null;
42
+ // Node registry for quick lookup
43
+ nodeRegistry = new Map();
44
+ constructor(options) {
45
+ this.options = options;
46
+ // Constructor sin effects - se sincronizan en las operaciones
47
+ }
48
+ // Data management
49
+ setData(data) {
50
+ this.nodeRegistry.clear();
51
+ const roots = this.buildNodes(data, null, 0);
52
+ this.roots.set(roots);
53
+ this.emitEvent({ type: 'initialized', treeModel: this });
54
+ }
55
+ buildNodes(data, parent, level) {
56
+ return data.map((item, index) => {
57
+ const node = new TreeNode(item, parent, level, index, this.options);
58
+ this.nodeRegistry.set(node.id, node);
59
+ // Recursively build children
60
+ const childrenData = item[this.options.childrenField || 'children'];
61
+ if (Array.isArray(childrenData) && childrenData.length > 0) {
62
+ node.children = this.buildNodes(childrenData, node, level + 1);
63
+ }
64
+ return node;
65
+ });
66
+ }
67
+ update() {
68
+ // Trigger change detection by creating new signal values
69
+ this.roots.set([...this.roots()]);
70
+ this.emitEvent({ type: 'update', treeModel: this });
71
+ }
72
+ // Node lookup
73
+ getNodeById(id) {
74
+ return this.nodeRegistry.get(id) || null;
75
+ }
76
+ getNodeBy(predicate) {
77
+ return this.getAllNodes().find(predicate) || null;
78
+ }
79
+ getAllNodes() {
80
+ const nodes = [];
81
+ const traverse = (nodeList) => {
82
+ nodeList.forEach(node => {
83
+ nodes.push(node);
84
+ if (node.children.length > 0) {
85
+ traverse(node.children);
86
+ }
87
+ });
88
+ };
89
+ traverse(this.roots());
90
+ return nodes;
91
+ }
92
+ // Navigation
93
+ focusNode(node) {
94
+ if (this.focusedNode()) {
95
+ this.focusedNode().setFocus(false);
96
+ }
97
+ if (node) {
98
+ this.focusedNodeId.set(node.id);
99
+ node.setFocus(true);
100
+ this.emitEvent({ type: 'focus', node });
101
+ }
102
+ else {
103
+ this.focusedNodeId.set(null);
104
+ }
105
+ }
106
+ focusNextNode() {
107
+ const flattened = this.flattenedNodes();
108
+ const currentIndex = flattened.findIndex(n => n.id === this.focusedNodeId());
109
+ if (currentIndex < flattened.length - 1) {
110
+ this.focusNode(flattened[currentIndex + 1]);
111
+ }
112
+ }
113
+ focusPreviousNode() {
114
+ const flattened = this.flattenedNodes();
115
+ const currentIndex = flattened.findIndex(n => n.id === this.focusedNodeId());
116
+ if (currentIndex > 0) {
117
+ this.focusNode(flattened[currentIndex - 1]);
118
+ }
119
+ }
120
+ focusDrillDown() {
121
+ const focused = this.focusedNode();
122
+ if (focused) {
123
+ if (!focused.isExpanded() && focused.hasChildren) {
124
+ focused.expand();
125
+ }
126
+ else if (focused.children.length > 0) {
127
+ this.focusNode(focused.children[0]);
128
+ }
129
+ }
130
+ }
131
+ focusDrillUp() {
132
+ const focused = this.focusedNode();
133
+ if (focused) {
134
+ if (focused.isExpanded() && focused.hasChildren) {
135
+ focused.collapse();
136
+ }
137
+ else if (focused.parent) {
138
+ this.focusNode(focused.parent);
139
+ }
140
+ }
141
+ }
142
+ // Operations
143
+ expandAll() {
144
+ const ids = new Set(this.getAllNodes().map(n => n.id));
145
+ this.expandedNodeIds.set(ids);
146
+ this.emitEvent({ type: 'expandAll', treeModel: this });
147
+ }
148
+ collapseAll() {
149
+ this.expandedNodeIds.set(new Set());
150
+ this.emitEvent({ type: 'collapseAll', treeModel: this });
151
+ }
152
+ expandNode(node) {
153
+ this.expandedNodeIds.update(ids => {
154
+ const newIds = new Set(ids);
155
+ newIds.add(node.id);
156
+ return newIds;
157
+ });
158
+ node.isExpanded.set(true);
159
+ this.emitEvent({ type: 'expand', node });
160
+ }
161
+ collapseNode(node) {
162
+ this.expandedNodeIds.update(ids => {
163
+ const newIds = new Set(ids);
164
+ newIds.delete(node.id);
165
+ return newIds;
166
+ });
167
+ node.isExpanded.set(false);
168
+ this.emitEvent({ type: 'collapse', node });
169
+ }
170
+ activateNode(node, multi = false) {
171
+ if (!multi) {
172
+ this.activeNodeIds.set(new Set([node.id]));
173
+ this.getAllNodes().forEach(n => n.isActive.set(n.id === node.id));
174
+ }
175
+ else {
176
+ this.activeNodeIds.update(ids => {
177
+ const newIds = new Set(ids);
178
+ newIds.add(node.id);
179
+ return newIds;
180
+ });
181
+ node.isActive.set(true);
182
+ }
183
+ this.emitEvent({ type: 'activate', node });
184
+ }
185
+ deactivateNode(node) {
186
+ this.activeNodeIds.update(ids => {
187
+ const newIds = new Set(ids);
188
+ newIds.delete(node.id);
189
+ return newIds;
190
+ });
191
+ node.isActive.set(false);
192
+ this.emitEvent({ type: 'deactivate', node });
193
+ }
194
+ selectNode(node, multi = false) {
195
+ if (!multi) {
196
+ this.selectedNodeIds.set(new Set([node.id]));
197
+ }
198
+ else {
199
+ this.selectedNodeIds.update(ids => {
200
+ const newIds = new Set(ids);
201
+ if (ids.has(node.id)) {
202
+ newIds.delete(node.id);
203
+ }
204
+ else {
205
+ newIds.add(node.id);
206
+ }
207
+ return newIds;
208
+ });
209
+ }
210
+ this.emitEvent({ type: 'select', node });
211
+ }
212
+ // Filtering
213
+ filterNodes(filterFn, autoShow = true) {
214
+ const hiddenIds = new Set();
215
+ this.getAllNodes().forEach(node => {
216
+ const matches = filterFn(node);
217
+ if (!matches) {
218
+ hiddenIds.add(node.id);
219
+ }
220
+ else if (autoShow) {
221
+ // Show ancestors of matching nodes
222
+ node.getAncestors().forEach(ancestor => {
223
+ hiddenIds.delete(ancestor.id);
224
+ });
225
+ }
226
+ });
227
+ this.hiddenNodeIds.set(hiddenIds);
228
+ this.emitEvent({ type: 'filter', treeModel: this });
229
+ }
230
+ clearFilter() {
231
+ this.hiddenNodeIds.set(new Set());
232
+ this.emitEvent({ type: 'clearFilter', treeModel: this });
233
+ }
234
+ // Drag & Drop
235
+ canMoveNode(node, to) {
236
+ // Prevent moving to itself or descendants
237
+ if (node === to.parent)
238
+ return false;
239
+ if (to.parent.getAncestors().includes(node))
240
+ return false;
241
+ // Custom validation
242
+ if (this.options.allowDrop) {
243
+ if (typeof this.options.allowDrop === 'function') {
244
+ return this.options.allowDrop(node, to);
245
+ }
246
+ return this.options.allowDrop;
247
+ }
248
+ return true;
249
+ }
250
+ moveNode(node, to) {
251
+ if (!this.canMoveNode(node, to))
252
+ return;
253
+ // Remove from old parent
254
+ if (node.parent) {
255
+ const oldIndex = node.parent.children.indexOf(node);
256
+ if (oldIndex !== -1) {
257
+ // Create new array to trigger change detection
258
+ node.parent.children = [
259
+ ...node.parent.children.slice(0, oldIndex),
260
+ ...node.parent.children.slice(oldIndex + 1)
261
+ ];
262
+ }
263
+ }
264
+ else {
265
+ // Handle root nodes
266
+ const rootIndex = this.roots().indexOf(node);
267
+ if (rootIndex !== -1) {
268
+ this.roots.set([
269
+ ...this.roots().slice(0, rootIndex),
270
+ ...this.roots().slice(rootIndex + 1)
271
+ ]);
272
+ }
273
+ }
274
+ // Add to new parent - create new array to trigger change detection
275
+ const newChildren = [...to.parent.children];
276
+ newChildren.splice(to.index, 0, node);
277
+ to.parent.children = newChildren;
278
+ node.parent = to.parent;
279
+ // Update levels
280
+ this.updateNodeLevels(node);
281
+ this.emitEvent({ type: 'moveNode', node, to });
282
+ this.update();
283
+ }
284
+ updateNodeLevels(node) {
285
+ const newLevel = node.parent ? node.parent.level + 1 : 0;
286
+ node.level = newLevel;
287
+ node.children.forEach(child => this.updateNodeLevels(child));
288
+ }
289
+ // Async children loading
290
+ async loadChildren(node) {
291
+ if (!this.options.getChildren)
292
+ return;
293
+ node.setLoading(true);
294
+ try {
295
+ const childrenData = await this.options.getChildren(node);
296
+ node.children = this.buildNodes(childrenData, node, node.level + 1);
297
+ node.hasChildren = node.children.length > 0;
298
+ this.emitEvent({ type: 'loadChildren', node });
299
+ this.update();
300
+ }
301
+ catch (error) {
302
+ this.emitEvent({ type: 'loadChildrenError', node, error });
303
+ }
304
+ finally {
305
+ node.setLoading(false);
306
+ }
307
+ }
308
+ // Events
309
+ emitEvent(event) {
310
+ this.events$.next(event);
311
+ }
312
+ }
313
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree.model.js","sourceRoot":"","sources":["../../../../../../../projects/concepto-user-controls/src/lib/concepto-tree/core/models/tree.model.ts"],"names":[],"mappings":"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","sourcesContent":["// lib/core/models/tree.model.ts\r\nimport { signal, computed, effect } from '@angular/core';\r\nimport { Subject } from 'rxjs';\r\nimport { TreeNode } from './tree-node.model';\r\nimport { TreeOptions } from './tree-options.model';\r\nimport { TreeEvent } from './tree-events.model';\r\n\r\nexport class TreeModel {\r\n  // Reactive state with signals\r\n  readonly roots = signal<TreeNode[]>([]);\r\n  readonly focusedNodeId = signal<string | number | null>(null);\r\n  readonly expandedNodeIds = signal<Set<string | number>>(new Set());\r\n  readonly activeNodeIds = signal<Set<string | number>>(new Set());\r\n  readonly selectedNodeIds = signal<Set<string | number>>(new Set());\r\n  readonly hiddenNodeIds = signal<Set<string | number>>(new Set());\r\n  \r\n  // Computed signals\r\n  readonly focusedNode = computed(() => {\r\n    const id = this.focusedNodeId();\r\n    return id ? this.getNodeById(id) : null;\r\n  });\r\n  \r\n  readonly expandedNodes = computed(() => \r\n    this.getAllNodes().filter(node => \r\n      this.expandedNodeIds().has(node.id)\r\n    )\r\n  );\r\n  \r\n  readonly activeNodes = computed(() =>\r\n    this.getAllNodes().filter(node =>\r\n      this.activeNodeIds().has(node.id)\r\n    )\r\n  );\r\n  \r\n  readonly selectedNodes = computed(() =>\r\n    this.getAllNodes().filter(node =>\r\n      this.selectedNodeIds().has(node.id)\r\n    )\r\n  );\r\n  \r\n  readonly visibleNodes = computed(() =>\r\n    this.getAllNodes().filter(node => !node.isHidden())\r\n  );\r\n  \r\n  readonly flattenedNodes = computed(() => {\r\n    const flattened: TreeNode[] = [];\r\n    const traverse = (nodes: TreeNode[]) => {\r\n      nodes.forEach(node => {\r\n        if (!node.isHidden()) {\r\n          flattened.push(node);\r\n          if (node.isExpanded() && node.children.length > 0) {\r\n            traverse(node.children);\r\n          }\r\n        }\r\n      });\r\n    };\r\n    traverse(this.roots());\r\n    return flattened;\r\n  });\r\n  \r\n  // Event streams\r\n  readonly events$ = new Subject<TreeEvent>();\r\n  \r\n  // Virtual root for unified handling\r\n  virtualRoot: TreeNode | null = null;\r\n  \r\n  // Node registry for quick lookup\r\n  private nodeRegistry = new Map<string | number, TreeNode>();\r\n  \r\n  constructor(public options: TreeOptions) {\r\n    // Constructor sin effects - se sincronizan en las operaciones\r\n  }\r\n  \r\n  // Data management\r\n  setData(data: any[]): void {\r\n    this.nodeRegistry.clear();\r\n    const roots = this.buildNodes(data, null, 0);\r\n    this.roots.set(roots);\r\n    this.emitEvent({ type: 'initialized', treeModel: this });\r\n  }\r\n  \r\n  private buildNodes(\r\n    data: any[],\r\n    parent: TreeNode | null,\r\n    level: number\r\n  ): TreeNode[] {\r\n    return data.map((item, index) => {\r\n      const node = new TreeNode(item, parent, level, index, this.options);\r\n      this.nodeRegistry.set(node.id, node);\r\n      \r\n      // Recursively build children\r\n      const childrenData = item[this.options.childrenField || 'children'];\r\n      if (Array.isArray(childrenData) && childrenData.length > 0) {\r\n        node.children = this.buildNodes(childrenData, node, level + 1);\r\n      }\r\n      \r\n      return node;\r\n    });\r\n  }\r\n  \r\n  update(): void {\r\n    // Trigger change detection by creating new signal values\r\n    this.roots.set([...this.roots()]);\r\n    this.emitEvent({ type: 'update', treeModel: this });\r\n  }\r\n  \r\n  // Node lookup\r\n  getNodeById(id: string | number): TreeNode | null {\r\n    return this.nodeRegistry.get(id) || null;\r\n  }\r\n  \r\n  getNodeBy(predicate: (node: TreeNode) => boolean): TreeNode | null {\r\n    return this.getAllNodes().find(predicate) || null;\r\n  }\r\n  \r\n  getAllNodes(): TreeNode[] {\r\n    const nodes: TreeNode[] = [];\r\n    const traverse = (nodeList: TreeNode[]) => {\r\n      nodeList.forEach(node => {\r\n        nodes.push(node);\r\n        if (node.children.length > 0) {\r\n          traverse(node.children);\r\n        }\r\n      });\r\n    };\r\n    traverse(this.roots());\r\n    return nodes;\r\n  }\r\n  \r\n  // Navigation\r\n  focusNode(node: TreeNode | null): void {\r\n    if (this.focusedNode()) {\r\n      this.focusedNode()!.setFocus(false);\r\n    }\r\n    if (node) {\r\n      this.focusedNodeId.set(node.id);\r\n      node.setFocus(true);\r\n      this.emitEvent({ type: 'focus', node });\r\n    } else {\r\n      this.focusedNodeId.set(null);\r\n    }\r\n  }\r\n  \r\n  focusNextNode(): void {\r\n    const flattened = this.flattenedNodes();\r\n    const currentIndex = flattened.findIndex(\r\n      n => n.id === this.focusedNodeId()\r\n    );\r\n    if (currentIndex < flattened.length - 1) {\r\n      this.focusNode(flattened[currentIndex + 1]);\r\n    }\r\n  }\r\n  \r\n  focusPreviousNode(): void {\r\n    const flattened = this.flattenedNodes();\r\n    const currentIndex = flattened.findIndex(\r\n      n => n.id === this.focusedNodeId()\r\n    );\r\n    if (currentIndex > 0) {\r\n      this.focusNode(flattened[currentIndex - 1]);\r\n    }\r\n  }\r\n  \r\n  focusDrillDown(): void {\r\n    const focused = this.focusedNode();\r\n    if (focused) {\r\n      if (!focused.isExpanded() && focused.hasChildren) {\r\n        focused.expand();\r\n      } else if (focused.children.length > 0) {\r\n        this.focusNode(focused.children[0]);\r\n      }\r\n    }\r\n  }\r\n  \r\n  focusDrillUp(): void {\r\n    const focused = this.focusedNode();\r\n    if (focused) {\r\n      if (focused.isExpanded() && focused.hasChildren) {\r\n        focused.collapse();\r\n      } else if (focused.parent) {\r\n        this.focusNode(focused.parent);\r\n      }\r\n    }\r\n  }\r\n  \r\n  // Operations\r\n  expandAll(): void {\r\n    const ids = new Set(this.getAllNodes().map(n => n.id));\r\n    this.expandedNodeIds.set(ids);\r\n    this.emitEvent({ type: 'expandAll', treeModel: this });\r\n  }\r\n  \r\n  collapseAll(): void {\r\n    this.expandedNodeIds.set(new Set());\r\n    this.emitEvent({ type: 'collapseAll', treeModel: this });\r\n  }\r\n  \r\n  expandNode(node: TreeNode): void {\r\n    this.expandedNodeIds.update(ids => {\r\n      const newIds = new Set(ids);\r\n      newIds.add(node.id);\r\n      return newIds;\r\n    });\r\n    node.isExpanded.set(true);\r\n    this.emitEvent({ type: 'expand', node });\r\n  }\r\n\r\n  collapseNode(node: TreeNode): void {\r\n    this.expandedNodeIds.update(ids => {\r\n      const newIds = new Set(ids);\r\n      newIds.delete(node.id);\r\n      return newIds;\r\n    });\r\n    node.isExpanded.set(false);\r\n    this.emitEvent({ type: 'collapse', node });\r\n  }\r\n\r\n  activateNode(node: TreeNode, multi: boolean = false): void {\r\n    if (!multi) {\r\n      this.activeNodeIds.set(new Set([node.id]));\r\n      this.getAllNodes().forEach(n => n.isActive.set(n.id === node.id));\r\n    } else {\r\n      this.activeNodeIds.update(ids => {\r\n        const newIds = new Set(ids);\r\n        newIds.add(node.id);\r\n        return newIds;\r\n      });\r\n      node.isActive.set(true);\r\n    }\r\n    this.emitEvent({ type: 'activate', node });\r\n  }\r\n  \r\n  deactivateNode(node: TreeNode): void {\r\n    this.activeNodeIds.update(ids => {\r\n      const newIds = new Set(ids);\r\n      newIds.delete(node.id);\r\n      return newIds;\r\n    });\r\n    node.isActive.set(false);\r\n    this.emitEvent({ type: 'deactivate', node });\r\n  }\r\n  \r\n  selectNode(node: TreeNode, multi: boolean = false): void {\r\n    if (!multi) {\r\n      this.selectedNodeIds.set(new Set([node.id]));\r\n    } else {\r\n      this.selectedNodeIds.update(ids => {\r\n        const newIds = new Set(ids);\r\n        if (ids.has(node.id)) {\r\n          newIds.delete(node.id);\r\n        } else {\r\n          newIds.add(node.id);\r\n        }\r\n        return newIds;\r\n      });\r\n    }\r\n    this.emitEvent({ type: 'select', node });\r\n  }\r\n  \r\n  // Filtering\r\n  filterNodes(filterFn: (node: TreeNode) => boolean, autoShow = true): void {\r\n    const hiddenIds = new Set<string | number>();\r\n    \r\n    this.getAllNodes().forEach(node => {\r\n      const matches = filterFn(node);\r\n      if (!matches) {\r\n        hiddenIds.add(node.id);\r\n      } else if (autoShow) {\r\n        // Show ancestors of matching nodes\r\n        node.getAncestors().forEach(ancestor => {\r\n          hiddenIds.delete(ancestor.id);\r\n        });\r\n      }\r\n    });\r\n    \r\n    this.hiddenNodeIds.set(hiddenIds);\r\n    this.emitEvent({ type: 'filter', treeModel: this });\r\n  }\r\n  \r\n  clearFilter(): void {\r\n    this.hiddenNodeIds.set(new Set());\r\n    this.emitEvent({ type: 'clearFilter', treeModel: this });\r\n  }\r\n  \r\n  // Drag & Drop\r\n  canMoveNode(\r\n    node: TreeNode,\r\n    to: { parent: TreeNode; index: number }\r\n  ): boolean {\r\n    // Prevent moving to itself or descendants\r\n    if (node === to.parent) return false;\r\n    if (to.parent.getAncestors().includes(node)) return false;\r\n    \r\n    // Custom validation\r\n    if (this.options.allowDrop) {\r\n      if (typeof this.options.allowDrop === 'function') {\r\n        return this.options.allowDrop(node, to);\r\n      }\r\n      return this.options.allowDrop;\r\n    }\r\n    \r\n    return true;\r\n  }\r\n  \r\n  moveNode(node: TreeNode, to: { parent: TreeNode; index: number }): void {\r\n    if (!this.canMoveNode(node, to)) return;\r\n\r\n    // Remove from old parent\r\n    if (node.parent) {\r\n      const oldIndex = node.parent.children.indexOf(node);\r\n      if (oldIndex !== -1) {\r\n        // Create new array to trigger change detection\r\n        node.parent.children = [\r\n          ...node.parent.children.slice(0, oldIndex),\r\n          ...node.parent.children.slice(oldIndex + 1)\r\n        ];\r\n      }\r\n    } else {\r\n      // Handle root nodes\r\n      const rootIndex = this.roots().indexOf(node);\r\n      if (rootIndex !== -1) {\r\n        this.roots.set([\r\n          ...this.roots().slice(0, rootIndex),\r\n          ...this.roots().slice(rootIndex + 1)\r\n        ]);\r\n      }\r\n    }\r\n\r\n    // Add to new parent - create new array to trigger change detection\r\n    const newChildren = [...to.parent.children];\r\n    newChildren.splice(to.index, 0, node);\r\n    to.parent.children = newChildren;\r\n    node.parent = to.parent;\r\n\r\n    // Update levels\r\n    this.updateNodeLevels(node);\r\n\r\n    this.emitEvent({ type: 'moveNode', node, to });\r\n    this.update();\r\n  }\r\n  \r\n  private updateNodeLevels(node: TreeNode): void {\r\n    const newLevel = node.parent ? node.parent.level + 1 : 0;\r\n    node.level = newLevel;\r\n    \r\n    node.children.forEach(child => this.updateNodeLevels(child));\r\n  }\r\n  \r\n  // Async children loading\r\n  async loadChildren(node: TreeNode): Promise<void> {\r\n    if (!this.options.getChildren) return;\r\n    \r\n    node.setLoading(true);\r\n    \r\n    try {\r\n      const childrenData = await this.options.getChildren(node);\r\n      node.children = this.buildNodes(childrenData, node, node.level + 1);\r\n      node.hasChildren = node.children.length > 0;\r\n      \r\n      this.emitEvent({ type: 'loadChildren', node });\r\n      this.update();\r\n    } catch (error) {\r\n      this.emitEvent({ type: 'loadChildrenError', node, error });\r\n    } finally {\r\n      node.setLoading(false);\r\n    }\r\n  }\r\n  \r\n  // Events\r\n  private emitEvent(event: TreeEvent): void {\r\n    this.events$.next(event);\r\n  }\r\n}"]}
@@ -0,0 +1,27 @@
1
+ // lib/core/services/tree-drag-drop.service.ts
2
+ import { Injectable, signal } from '@angular/core';
3
+ import * as i0 from "@angular/core";
4
+ export class TreeDragDropService {
5
+ draggedNode = signal(null);
6
+ setDraggedNode(node) {
7
+ this.draggedNode.set(node);
8
+ }
9
+ getDraggedNode() {
10
+ return this.draggedNode();
11
+ }
12
+ clearDraggedNode() {
13
+ this.draggedNode.set(null);
14
+ }
15
+ isDragging() {
16
+ return this.draggedNode() !== null;
17
+ }
18
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeDragDropService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
19
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeDragDropService, providedIn: 'root' });
20
+ }
21
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeDragDropService, decorators: [{
22
+ type: Injectable,
23
+ args: [{
24
+ providedIn: 'root',
25
+ }]
26
+ }] });
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJlZS1kcmFnLWRyb3Auc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbmNlcHRvLXVzZXItY29udHJvbHMvc3JjL2xpYi9jb25jZXB0by10cmVlL2NvcmUvc2VydmljZXMvdHJlZS1kcmFnLWRyb3Auc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSw4Q0FBOEM7QUFDOUMsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBTW5ELE1BQU0sT0FBTyxtQkFBbUI7SUFDYixXQUFXLEdBQUcsTUFBTSxDQUFrQixJQUFJLENBQUMsQ0FBQztJQUU5RCxjQUFjLENBQUMsSUFBYztRQUMxQixJQUFJLENBQUMsV0FBVyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUM3QixDQUFDO0lBRUQsY0FBYztRQUNaLE9BQU8sSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO0lBQzVCLENBQUM7SUFFRCxnQkFBZ0I7UUFDZCxJQUFJLENBQUMsV0FBVyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUM3QixDQUFDO0lBRUQsVUFBVTtRQUNSLE9BQU8sSUFBSSxDQUFDLFdBQVcsRUFBRSxLQUFLLElBQUksQ0FBQztJQUNyQyxDQUFDO3dHQWpCVSxtQkFBbUI7NEdBQW5CLG1CQUFtQixjQUZsQixNQUFNOzs0RkFFUCxtQkFBbUI7a0JBSC9CLFVBQVU7bUJBQUM7b0JBQ1YsVUFBVSxFQUFFLE1BQU07aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiLy8gbGliL2NvcmUvc2VydmljZXMvdHJlZS1kcmFnLWRyb3Auc2VydmljZS50c1xyXG5pbXBvcnQgeyBJbmplY3RhYmxlLCBzaWduYWwgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgVHJlZU5vZGUgfSBmcm9tICcuLi9tb2RlbHMvdHJlZS1ub2RlLm1vZGVsJztcclxuXHJcbkBJbmplY3RhYmxlKHtcclxuICBwcm92aWRlZEluOiAncm9vdCcsXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBUcmVlRHJhZ0Ryb3BTZXJ2aWNlIHtcclxuICBwcml2YXRlIHJlYWRvbmx5IGRyYWdnZWROb2RlID0gc2lnbmFsPFRyZWVOb2RlIHwgbnVsbD4obnVsbCk7XHJcbiAgXHJcbiBzZXREcmFnZ2VkTm9kZShub2RlOiBUcmVlTm9kZSk6IHZvaWQge1xyXG4gICAgdGhpcy5kcmFnZ2VkTm9kZS5zZXQobm9kZSk7XHJcbiAgfVxyXG4gIFxyXG4gIGdldERyYWdnZWROb2RlKCk6IFRyZWVOb2RlIHwgbnVsbCB7XHJcbiAgICByZXR1cm4gdGhpcy5kcmFnZ2VkTm9kZSgpO1xyXG4gIH1cclxuICBcclxuICBjbGVhckRyYWdnZWROb2RlKCk6IHZvaWQge1xyXG4gICAgdGhpcy5kcmFnZ2VkTm9kZS5zZXQobnVsbCk7XHJcbiAgfVxyXG4gIFxyXG4gIGlzRHJhZ2dpbmcoKTogYm9vbGVhbiB7XHJcbiAgICByZXR1cm4gdGhpcy5kcmFnZ2VkTm9kZSgpICE9PSBudWxsO1xyXG4gIH1cclxufSJdfQ==
@@ -0,0 +1,69 @@
1
+ // lib/directives/tree-drag.directive.ts
2
+ import { Directive, Input, HostListener, HostBinding, ElementRef, inject, // Add inject
3
+ } from '@angular/core';
4
+ import { TreeDragDropService } from '../core/services/tree-drag-drop.service';
5
+ import * as i0 from "@angular/core";
6
+ export class TreeDragDirective {
7
+ node;
8
+ treeModel;
9
+ options;
10
+ // Use inject() function instead of constructor injection
11
+ elementRef = inject(ElementRef);
12
+ dragDropService = inject(TreeDragDropService);
13
+ get draggable() {
14
+ if (typeof this.options.allowDrag === 'function') {
15
+ return this.options.allowDrag(this.node);
16
+ }
17
+ return this.options.allowDrag || false;
18
+ }
19
+ onDragStart(event) {
20
+ if (!this.draggable) {
21
+ event.preventDefault();
22
+ return;
23
+ }
24
+ this.dragDropService.setDraggedNode(this.node);
25
+ if (event.dataTransfer) {
26
+ event.dataTransfer.effectAllowed = 'move';
27
+ event.dataTransfer.setData('text/plain', this.node.id.toString());
28
+ // Create drag image
29
+ const dragImage = this.elementRef.nativeElement.cloneNode(true);
30
+ dragImage.style.opacity = '0.7';
31
+ document.body.appendChild(dragImage);
32
+ event.dataTransfer.setDragImage(dragImage, 0, 0);
33
+ setTimeout(() => document.body.removeChild(dragImage), 0);
34
+ }
35
+ this.elementRef.nativeElement.classList.add('tree-node-dragging');
36
+ }
37
+ onDragEnd(event) {
38
+ this.dragDropService.clearDraggedNode();
39
+ this.elementRef.nativeElement.classList.remove('tree-node-dragging');
40
+ }
41
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeDragDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
42
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: TreeDragDirective, isStandalone: true, selector: "[treeDrag]", inputs: { node: ["treeDrag", "node"], treeModel: "treeModel", options: "options" }, host: { listeners: { "dragstart": "onDragStart($event)", "dragend": "onDragEnd($event)" }, properties: { "attr.draggable": "this.draggable" } }, ngImport: i0 });
43
+ }
44
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeDragDirective, decorators: [{
45
+ type: Directive,
46
+ args: [{
47
+ selector: '[treeDrag]',
48
+ standalone: true,
49
+ }]
50
+ }], propDecorators: { node: [{
51
+ type: Input,
52
+ args: [{ required: true, alias: 'treeDrag' }]
53
+ }], treeModel: [{
54
+ type: Input,
55
+ args: [{ required: true }]
56
+ }], options: [{
57
+ type: Input,
58
+ args: [{ required: true }]
59
+ }], draggable: [{
60
+ type: HostBinding,
61
+ args: ['attr.draggable']
62
+ }], onDragStart: [{
63
+ type: HostListener,
64
+ args: ['dragstart', ['$event']]
65
+ }], onDragEnd: [{
66
+ type: HostListener,
67
+ args: ['dragend', ['$event']]
68
+ }] } });
69
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJlZS1kcmFnLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbmNlcHRvLXVzZXItY29udHJvbHMvc3JjL2xpYi9jb25jZXB0by10cmVlL2RpcmVjdGl2ZXMvdHJlZS1kcmFnLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSx3Q0FBd0M7QUFDeEMsT0FBTyxFQUNMLFNBQVMsRUFDVCxLQUFLLEVBQ0wsWUFBWSxFQUNaLFdBQVcsRUFDWCxVQUFVLEVBQ1YsTUFBTSxFQUFFLGFBQWE7RUFDdEIsTUFBTSxlQUFlLENBQUM7QUFJdkIsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0seUNBQXlDLENBQUM7O0FBTTlFLE1BQU0sT0FBTyxpQkFBaUI7SUFDa0IsSUFBSSxDQUFZO0lBQ25DLFNBQVMsQ0FBYTtJQUN0QixPQUFPLENBQWU7SUFFakQseURBQXlEO0lBQ3hDLFVBQVUsR0FBRyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDaEMsZUFBZSxHQUFHLE1BQU0sQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO0lBRS9ELElBQ0ksU0FBUztRQUNYLElBQUksT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFDLFNBQVMsS0FBSyxVQUFVLEVBQUUsQ0FBQztZQUNqRCxPQUFPLElBQUksQ0FBQyxPQUFPLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUMzQyxDQUFDO1FBQ0QsT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFDLFNBQVMsSUFBSSxLQUFLLENBQUM7SUFDekMsQ0FBQztJQUdELFdBQVcsQ0FBQyxLQUFnQjtRQUMxQixJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO1lBQ3BCLEtBQUssQ0FBQyxjQUFjLEVBQUUsQ0FBQztZQUN2QixPQUFPO1FBQ1QsQ0FBQztRQUVELElBQUksQ0FBQyxlQUFlLENBQUMsY0FBYyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUUvQyxJQUFJLEtBQUssQ0FBQyxZQUFZLEVBQUUsQ0FBQztZQUN2QixLQUFLLENBQUMsWUFBWSxDQUFDLGFBQWEsR0FBRyxNQUFNLENBQUM7WUFDMUMsS0FBSyxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsWUFBWSxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUM7WUFFbEUsb0JBQW9CO1lBQ3BCLE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQWdCLENBQUM7WUFDL0UsU0FBUyxDQUFDLEtBQUssQ0FBQyxPQUFPLEdBQUcsS0FBSyxDQUFDO1lBQ2hDLFFBQVEsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQ3JDLEtBQUssQ0FBQyxZQUFZLENBQUMsWUFBWSxDQUFDLFNBQVMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7WUFFakQsVUFBVSxDQUFDLEdBQUcsRUFBRSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLFNBQVMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQzVELENBQUM7UUFFRCxJQUFJLENBQUMsVUFBVSxDQUFDLGFBQWEsQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLG9CQUFvQixDQUFDLENBQUM7SUFDcEUsQ0FBQztJQUdELFNBQVMsQ0FBQyxLQUFnQjtRQUN4QixJQUFJLENBQUMsZUFBZSxDQUFDLGdCQUFnQixFQUFFLENBQUM7UUFDeEMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDO0lBQ3ZFLENBQUM7d0dBOUNVLGlCQUFpQjs0RkFBakIsaUJBQWlCOzs0RkFBakIsaUJBQWlCO2tCQUo3QixTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSxZQUFZO29CQUN0QixVQUFVLEVBQUUsSUFBSTtpQkFDakI7OEJBRStDLElBQUk7c0JBQWpELEtBQUs7dUJBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLEtBQUssRUFBRSxVQUFVLEVBQUU7Z0JBQ2pCLFNBQVM7c0JBQW5DLEtBQUs7dUJBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFO2dCQUNFLE9BQU87c0JBQWpDLEtBQUs7dUJBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFO2dCQU9yQixTQUFTO3NCQURaLFdBQVc7dUJBQUMsZ0JBQWdCO2dCQVM3QixXQUFXO3NCQURWLFlBQVk7dUJBQUMsV0FBVyxFQUFFLENBQUMsUUFBUSxDQUFDO2dCQTBCckMsU0FBUztzQkFEUixZQUFZO3VCQUFDLFNBQVMsRUFBRSxDQUFDLFFBQVEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIGxpYi9kaXJlY3RpdmVzL3RyZWUtZHJhZy5kaXJlY3RpdmUudHNcclxuaW1wb3J0IHtcclxuICBEaXJlY3RpdmUsXHJcbiAgSW5wdXQsXHJcbiAgSG9zdExpc3RlbmVyLFxyXG4gIEhvc3RCaW5kaW5nLFxyXG4gIEVsZW1lbnRSZWYsXHJcbiAgaW5qZWN0LCAvLyBBZGQgaW5qZWN0XHJcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IFRyZWVOb2RlIH0gZnJvbSAnLi4vY29yZS9tb2RlbHMvdHJlZS1ub2RlLm1vZGVsJztcclxuaW1wb3J0IHsgVHJlZU1vZGVsIH0gZnJvbSAnLi4vY29yZS9tb2RlbHMvdHJlZS5tb2RlbCc7XHJcbmltcG9ydCB7IFRyZWVPcHRpb25zIH0gZnJvbSAnLi4vY29yZS9tb2RlbHMvdHJlZS1vcHRpb25zLm1vZGVsJztcclxuaW1wb3J0IHsgVHJlZURyYWdEcm9wU2VydmljZSB9IGZyb20gJy4uL2NvcmUvc2VydmljZXMvdHJlZS1kcmFnLWRyb3Auc2VydmljZSc7XHJcblxyXG5ARGlyZWN0aXZlKHtcclxuICBzZWxlY3RvcjogJ1t0cmVlRHJhZ10nLFxyXG4gIHN0YW5kYWxvbmU6IHRydWUsXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBUcmVlRHJhZ0RpcmVjdGl2ZSB7XHJcbiAgQElucHV0KHsgcmVxdWlyZWQ6IHRydWUsIGFsaWFzOiAndHJlZURyYWcnIH0pIG5vZGUhOiBUcmVlTm9kZTtcclxuICBASW5wdXQoeyByZXF1aXJlZDogdHJ1ZSB9KSB0cmVlTW9kZWwhOiBUcmVlTW9kZWw7XHJcbiAgQElucHV0KHsgcmVxdWlyZWQ6IHRydWUgfSkgb3B0aW9ucyE6IFRyZWVPcHRpb25zO1xyXG4gIFxyXG4gIC8vIFVzZSBpbmplY3QoKSBmdW5jdGlvbiBpbnN0ZWFkIG9mIGNvbnN0cnVjdG9yIGluamVjdGlvblxyXG4gIHByaXZhdGUgcmVhZG9ubHkgZWxlbWVudFJlZiA9IGluamVjdChFbGVtZW50UmVmKTtcclxuICBwcml2YXRlIHJlYWRvbmx5IGRyYWdEcm9wU2VydmljZSA9IGluamVjdChUcmVlRHJhZ0Ryb3BTZXJ2aWNlKTtcclxuICBcclxuICBASG9zdEJpbmRpbmcoJ2F0dHIuZHJhZ2dhYmxlJylcclxuICBnZXQgZHJhZ2dhYmxlKCk6IGJvb2xlYW4ge1xyXG4gICAgaWYgKHR5cGVvZiB0aGlzLm9wdGlvbnMuYWxsb3dEcmFnID09PSAnZnVuY3Rpb24nKSB7XHJcbiAgICAgIHJldHVybiB0aGlzLm9wdGlvbnMuYWxsb3dEcmFnKHRoaXMubm9kZSk7XHJcbiAgICB9XHJcbiAgICByZXR1cm4gdGhpcy5vcHRpb25zLmFsbG93RHJhZyB8fCBmYWxzZTtcclxuICB9XHJcbiAgXHJcbiAgQEhvc3RMaXN0ZW5lcignZHJhZ3N0YXJ0JywgWyckZXZlbnQnXSlcclxuICBvbkRyYWdTdGFydChldmVudDogRHJhZ0V2ZW50KTogdm9pZCB7XHJcbiAgICBpZiAoIXRoaXMuZHJhZ2dhYmxlKSB7XHJcbiAgICAgIGV2ZW50LnByZXZlbnREZWZhdWx0KCk7XHJcbiAgICAgIHJldHVybjtcclxuICAgIH1cclxuICAgIFxyXG4gICAgdGhpcy5kcmFnRHJvcFNlcnZpY2Uuc2V0RHJhZ2dlZE5vZGUodGhpcy5ub2RlKTtcclxuICAgIFxyXG4gICAgaWYgKGV2ZW50LmRhdGFUcmFuc2Zlcikge1xyXG4gICAgICBldmVudC5kYXRhVHJhbnNmZXIuZWZmZWN0QWxsb3dlZCA9ICdtb3ZlJztcclxuICAgICAgZXZlbnQuZGF0YVRyYW5zZmVyLnNldERhdGEoJ3RleHQvcGxhaW4nLCB0aGlzLm5vZGUuaWQudG9TdHJpbmcoKSk7XHJcbiAgICAgIFxyXG4gICAgICAvLyBDcmVhdGUgZHJhZyBpbWFnZVxyXG4gICAgICBjb25zdCBkcmFnSW1hZ2UgPSB0aGlzLmVsZW1lbnRSZWYubmF0aXZlRWxlbWVudC5jbG9uZU5vZGUodHJ1ZSkgYXMgSFRNTEVsZW1lbnQ7XHJcbiAgICAgIGRyYWdJbWFnZS5zdHlsZS5vcGFjaXR5ID0gJzAuNyc7XHJcbiAgICAgIGRvY3VtZW50LmJvZHkuYXBwZW5kQ2hpbGQoZHJhZ0ltYWdlKTtcclxuICAgICAgZXZlbnQuZGF0YVRyYW5zZmVyLnNldERyYWdJbWFnZShkcmFnSW1hZ2UsIDAsIDApO1xyXG4gICAgICBcclxuICAgICAgc2V0VGltZW91dCgoKSA9PiBkb2N1bWVudC5ib2R5LnJlbW92ZUNoaWxkKGRyYWdJbWFnZSksIDApO1xyXG4gICAgfVxyXG4gICAgXHJcbiAgICB0aGlzLmVsZW1lbnRSZWYubmF0aXZlRWxlbWVudC5jbGFzc0xpc3QuYWRkKCd0cmVlLW5vZGUtZHJhZ2dpbmcnKTtcclxuICB9XHJcbiAgXHJcbiAgQEhvc3RMaXN0ZW5lcignZHJhZ2VuZCcsIFsnJGV2ZW50J10pXHJcbiAgb25EcmFnRW5kKGV2ZW50OiBEcmFnRXZlbnQpOiB2b2lkIHtcclxuICAgIHRoaXMuZHJhZ0Ryb3BTZXJ2aWNlLmNsZWFyRHJhZ2dlZE5vZGUoKTtcclxuICAgIHRoaXMuZWxlbWVudFJlZi5uYXRpdmVFbGVtZW50LmNsYXNzTGlzdC5yZW1vdmUoJ3RyZWUtbm9kZS1kcmFnZ2luZycpO1xyXG4gIH1cclxufSJdfQ==
@@ -0,0 +1,124 @@
1
+ // lib/directives/tree-drop.directive.ts
2
+ import { Directive, Input, HostListener, HostBinding, ElementRef, signal, inject, // Add inject
3
+ } from '@angular/core';
4
+ import { TreeDragDropService } from '../core/services/tree-drag-drop.service';
5
+ import * as i0 from "@angular/core";
6
+ export class TreeDropDirective {
7
+ node;
8
+ treeModel;
9
+ options;
10
+ dropPosition = 'inside';
11
+ // Use inject() function instead of constructor injection
12
+ elementRef = inject(ElementRef);
13
+ dragDropService = inject(TreeDragDropService);
14
+ isDraggingOver = signal(false);
15
+ canDrop = signal(false);
16
+ get isDropTarget() {
17
+ return this.isDraggingOver() && this.canDrop();
18
+ }
19
+ get isDropDisabled() {
20
+ return this.isDraggingOver() && !this.canDrop();
21
+ }
22
+ onDragEnter(event) {
23
+ event.preventDefault();
24
+ const draggedNode = this.dragDropService.getDraggedNode();
25
+ if (!draggedNode)
26
+ return;
27
+ this.isDraggingOver.set(true);
28
+ const dropTarget = this.getDropTarget();
29
+ const canDrop = this.treeModel.canMoveNode(draggedNode, dropTarget);
30
+ this.canDrop.set(canDrop);
31
+ if (canDrop && event.dataTransfer) {
32
+ event.dataTransfer.dropEffect = 'move';
33
+ }
34
+ }
35
+ onDragOver(event) {
36
+ if (this.canDrop()) {
37
+ event.preventDefault();
38
+ if (event.dataTransfer) {
39
+ event.dataTransfer.dropEffect = 'move';
40
+ }
41
+ }
42
+ }
43
+ onDragLeave(event) {
44
+ // Check if we're really leaving (not just entering a child element)
45
+ const rect = this.elementRef.nativeElement.getBoundingClientRect();
46
+ const x = event.clientX;
47
+ const y = event.clientY;
48
+ if (x < rect.left || x >= rect.right || y < rect.top || y >= rect.bottom) {
49
+ this.isDraggingOver.set(false);
50
+ this.canDrop.set(false);
51
+ }
52
+ }
53
+ onDrop(event) {
54
+ event.preventDefault();
55
+ event.stopPropagation();
56
+ const draggedNode = this.dragDropService.getDraggedNode();
57
+ if (!draggedNode || !this.canDrop())
58
+ return;
59
+ const dropTarget = this.getDropTarget();
60
+ this.treeModel.moveNode(draggedNode, dropTarget);
61
+ this.isDraggingOver.set(false);
62
+ this.canDrop.set(false);
63
+ this.dragDropService.clearDraggedNode();
64
+ }
65
+ getDropTarget() {
66
+ switch (this.dropPosition) {
67
+ case 'before':
68
+ return {
69
+ parent: this.node.parent,
70
+ index: this.node.index,
71
+ };
72
+ case 'after':
73
+ return {
74
+ parent: this.node.parent,
75
+ index: this.node.index + 1,
76
+ };
77
+ case 'inside':
78
+ default:
79
+ return {
80
+ parent: this.node,
81
+ index: this.node.children.length,
82
+ };
83
+ }
84
+ }
85
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeDropDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
86
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: TreeDropDirective, isStandalone: true, selector: "[treeDrop]", inputs: { node: ["treeDrop", "node"], treeModel: "treeModel", options: "options", dropPosition: "dropPosition" }, host: { listeners: { "dragenter": "onDragEnter($event)", "dragover": "onDragOver($event)", "dragleave": "onDragLeave($event)", "drop": "onDrop($event)" }, properties: { "class.tree-drop-target": "this.isDropTarget", "class.tree-drop-disabled": "this.isDropDisabled" } }, ngImport: i0 });
87
+ }
88
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeDropDirective, decorators: [{
89
+ type: Directive,
90
+ args: [{
91
+ selector: '[treeDrop]',
92
+ standalone: true,
93
+ }]
94
+ }], propDecorators: { node: [{
95
+ type: Input,
96
+ args: [{ required: true, alias: 'treeDrop' }]
97
+ }], treeModel: [{
98
+ type: Input,
99
+ args: [{ required: true }]
100
+ }], options: [{
101
+ type: Input,
102
+ args: [{ required: true }]
103
+ }], dropPosition: [{
104
+ type: Input
105
+ }], isDropTarget: [{
106
+ type: HostBinding,
107
+ args: ['class.tree-drop-target']
108
+ }], isDropDisabled: [{
109
+ type: HostBinding,
110
+ args: ['class.tree-drop-disabled']
111
+ }], onDragEnter: [{
112
+ type: HostListener,
113
+ args: ['dragenter', ['$event']]
114
+ }], onDragOver: [{
115
+ type: HostListener,
116
+ args: ['dragover', ['$event']]
117
+ }], onDragLeave: [{
118
+ type: HostListener,
119
+ args: ['dragleave', ['$event']]
120
+ }], onDrop: [{
121
+ type: HostListener,
122
+ args: ['drop', ['$event']]
123
+ }] } });
124
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree-drop.directive.js","sourceRoot":"","sources":["../../../../../../projects/concepto-user-controls/src/lib/concepto-tree/directives/tree-drop.directive.ts"],"names":[],"mappings":"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","sourcesContent":["// lib/directives/tree-drop.directive.ts\r\nimport {\r\n  Directive,\r\n  Input,\r\n  HostListener,\r\n  HostBinding,\r\n  ElementRef,\r\n  signal,\r\n  inject, // Add inject\r\n} from '@angular/core';\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 { TreeDragDropService } from '../core/services/tree-drag-drop.service';\r\n\r\ntype DropPosition = 'before' | 'after' | 'inside';\r\n\r\n@Directive({\r\n  selector: '[treeDrop]',\r\n  standalone: true,\r\n})\r\nexport class TreeDropDirective {\r\n  @Input({ required: true, alias: 'treeDrop' }) node!: TreeNode;\r\n  @Input({ required: true }) treeModel!: TreeModel;\r\n  @Input({ required: true }) options!: TreeOptions;\r\n  @Input() dropPosition: DropPosition = 'inside';\r\n  \r\n  // Use inject() function instead of constructor injection\r\n  private readonly elementRef = inject(ElementRef);\r\n  private readonly dragDropService = inject(TreeDragDropService);\r\n  \r\n  readonly isDraggingOver = signal(false);\r\n  readonly canDrop = signal(false);\r\n  \r\n  @HostBinding('class.tree-drop-target')\r\n  get isDropTarget(): boolean {\r\n    return this.isDraggingOver() && this.canDrop();\r\n  }\r\n  \r\n  @HostBinding('class.tree-drop-disabled')\r\n  get isDropDisabled(): boolean {\r\n    return this.isDraggingOver() && !this.canDrop();\r\n  }\r\n  \r\n  @HostListener('dragenter', ['$event'])\r\n  onDragEnter(event: DragEvent): void {\r\n    event.preventDefault();\r\n    \r\n    const draggedNode = this.dragDropService.getDraggedNode();\r\n    if (!draggedNode) return;\r\n    \r\n    this.isDraggingOver.set(true);\r\n    \r\n    const dropTarget = this.getDropTarget();\r\n    const canDrop = this.treeModel.canMoveNode(draggedNode, dropTarget);\r\n    this.canDrop.set(canDrop);\r\n    \r\n    if (canDrop && event.dataTransfer) {\r\n      event.dataTransfer.dropEffect = 'move';\r\n    }\r\n  }\r\n  \r\n  @HostListener('dragover', ['$event'])\r\n  onDragOver(event: DragEvent): void {\r\n    if (this.canDrop()) {\r\n      event.preventDefault();\r\n      if (event.dataTransfer) {\r\n        event.dataTransfer.dropEffect = 'move';\r\n      }\r\n    }\r\n  }\r\n  \r\n  @HostListener('dragleave', ['$event'])\r\n  onDragLeave(event: DragEvent): void {\r\n    // Check if we're really leaving (not just entering a child element)\r\n    const rect = this.elementRef.nativeElement.getBoundingClientRect();\r\n    const x = event.clientX;\r\n    const y = event.clientY;\r\n    \r\n    if (x < rect.left || x >= rect.right || y < rect.top || y >= rect.bottom) {\r\n      this.isDraggingOver.set(false);\r\n      this.canDrop.set(false);\r\n    }\r\n  }\r\n  \r\n  @HostListener('drop', ['$event'])\r\n  onDrop(event: DragEvent): void {\r\n    event.preventDefault();\r\n    event.stopPropagation();\r\n    \r\n    const draggedNode = this.dragDropService.getDraggedNode();\r\n    if (!draggedNode || !this.canDrop()) return;\r\n    \r\n    const dropTarget = this.getDropTarget();\r\n    this.treeModel.moveNode(draggedNode, dropTarget);\r\n    \r\n    this.isDraggingOver.set(false);\r\n    this.canDrop.set(false);\r\n    this.dragDropService.clearDraggedNode();\r\n  }\r\n  \r\n  private getDropTarget(): { parent: TreeNode; index: number } {\r\n    switch (this.dropPosition) {\r\n      case 'before':\r\n        return {\r\n          parent: this.node.parent!,\r\n          index: this.node.index,\r\n        };\r\n      case 'after':\r\n        return {\r\n          parent: this.node.parent!,\r\n          index: this.node.index + 1,\r\n        };\r\n      case 'inside':\r\n      default:\r\n        return {\r\n          parent: this.node,\r\n          index: this.node.children.length,\r\n        };\r\n    }\r\n  }\r\n}"]}
@@ -0,0 +1,19 @@
1
+ // lib/directives/tree-node-template.directive.ts
2
+ import { Directive } from '@angular/core';
3
+ import * as i0 from "@angular/core";
4
+ export class TreeNodeTemplateDirective {
5
+ template;
6
+ constructor(template) {
7
+ this.template = template;
8
+ }
9
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeNodeTemplateDirective, deps: [{ token: i0.TemplateRef }], target: i0.ɵɵFactoryTarget.Directive });
10
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: TreeNodeTemplateDirective, isStandalone: true, selector: "[treeNodeTemplate]", ngImport: i0 });
11
+ }
12
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TreeNodeTemplateDirective, decorators: [{
13
+ type: Directive,
14
+ args: [{
15
+ selector: '[treeNodeTemplate]',
16
+ standalone: true,
17
+ }]
18
+ }], ctorParameters: () => [{ type: i0.TemplateRef }] });
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJlZS1ub2RlLXRlbXBsYXRlLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbmNlcHRvLXVzZXItY29udHJvbHMvc3JjL2xpYi9jb25jZXB0by10cmVlL2RpcmVjdGl2ZXMvdHJlZS1ub2RlLXRlbXBsYXRlLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxpREFBaUQ7QUFDakQsT0FBTyxFQUFFLFNBQVMsRUFBZSxNQUFNLGVBQWUsQ0FBQzs7QUFNdkQsTUFBTSxPQUFPLHlCQUF5QjtJQUNqQjtJQUFuQixZQUFtQixRQUEwQjtRQUExQixhQUFRLEdBQVIsUUFBUSxDQUFrQjtJQUFHLENBQUM7d0dBRHRDLHlCQUF5Qjs0RkFBekIseUJBQXlCOzs0RkFBekIseUJBQXlCO2tCQUpyQyxTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSxvQkFBb0I7b0JBQzlCLFVBQVUsRUFBRSxJQUFJO2lCQUNqQiIsInNvdXJjZXNDb250ZW50IjpbIi8vIGxpYi9kaXJlY3RpdmVzL3RyZWUtbm9kZS10ZW1wbGF0ZS5kaXJlY3RpdmUudHNcclxuaW1wb3J0IHsgRGlyZWN0aXZlLCBUZW1wbGF0ZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5cclxuQERpcmVjdGl2ZSh7XHJcbiAgc2VsZWN0b3I6ICdbdHJlZU5vZGVUZW1wbGF0ZV0nLFxyXG4gIHN0YW5kYWxvbmU6IHRydWUsXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBUcmVlTm9kZVRlbXBsYXRlRGlyZWN0aXZlIHtcclxuICBjb25zdHJ1Y3RvcihwdWJsaWMgdGVtcGxhdGU6IFRlbXBsYXRlUmVmPGFueT4pIHt9XHJcbn0iXX0=