@odymaui/angular-tree-component
Version:
A simple yet powerful tree component for Angular16. WARNING: This is an unsupported fork for use in a dependent project to upgrade it to Angular 16. Unit tests pass and the example-app works as expected.
602 lines (596 loc) • 58.6 kB
JavaScript
import { Component, ContentChild, EventEmitter, HostListener, Input, Output, ViewChild, ViewEncapsulation } from '@angular/core';
import { TreeModel } from '../models/tree.model';
import { makeObservable, reaction } from 'mobx';
import { observable, computed, action } from '../mobx-angular/mobx-proxy';
import * as i0 from "@angular/core";
import * as i1 from "../models/tree.model";
import * as i2 from "../models/tree-dragged-element.model";
import * as i3 from "@angular/common";
import * as i4 from "./tree-node-drop-slot.component";
import * as i5 from "./tree-viewport.component";
import * as i6 from "./loading.component";
import * as i7 from "../directives/tree-animate-open.directive";
import * as i8 from "../mobx-angular/tree-mobx-autorun.directive";
import * as i9 from "../directives/tree-drop.directive";
import * as i10 from "../directives/tree-drag.directive";
import * as i11 from "./tree-node-expander.component";
import * as i12 from "./tree-node-checkbox.component";
export class TreeComponent {
treeModel;
treeDraggedElement;
_nodes;
_options;
loadingTemplate;
treeNodeTemplate;
treeNodeWrapperTemplate;
treeNodeFullTemplate;
viewportComponent;
// Will be handled in ngOnChanges
set nodes(nodes) {
}
;
set options(options) {
}
;
set focused(value) {
this.treeModel.setFocus(value);
}
set state(state) {
this.treeModel.setState(state);
}
toggleExpanded;
activate;
deactivate;
nodeActivate;
nodeDeactivate;
select;
deselect;
focus;
blur;
updateData;
initialized;
moveNode;
copyNode;
loadNodeChildren;
changeFilter;
event;
stateChange;
constructor(treeModel, treeDraggedElement) {
this.treeModel = treeModel;
this.treeDraggedElement = treeDraggedElement;
treeModel.eventNames.forEach((name) => this[name] = new EventEmitter());
treeModel.subscribeToState((state) => this.stateChange.emit(state));
}
onKeydown($event) {
if (!this.treeModel.isFocused)
return;
if (['input', 'textarea'].includes(document.activeElement.tagName.toLowerCase()))
return;
const focusedNode = this.treeModel.getFocusedNode();
this.treeModel.performKeyAction(focusedNode, $event);
}
onMousedown($event) {
function isOutsideClick(startElement, nodeName) {
return !startElement ? true : startElement.localName === nodeName ? false : isOutsideClick(startElement.parentElement, nodeName);
}
if (isOutsideClick($event.target, 'tree-root')) {
this.treeModel.setFocus(false);
}
}
ngOnChanges(changes) {
if (changes.options || changes.nodes) {
this.treeModel.setData({
options: changes.options && changes.options.currentValue,
nodes: changes.nodes && changes.nodes.currentValue,
events: this.pick(this, this.treeModel.eventNames)
});
}
}
sizeChanged() {
this.viewportComponent.setViewport();
}
pick(object, keys) {
return keys.reduce((obj, key) => {
if (object && object.hasOwnProperty(key)) {
obj[key] = object[key];
}
return obj;
}, {});
}
/** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeComponent, deps: [{ token: i1.TreeModel }, { token: i2.TreeDraggedElement }], target: i0.ɵɵFactoryTarget.Component });
/** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.2", type: TreeComponent, selector: "Tree, tree-root", inputs: { nodes: "nodes", options: "options", focused: "focused", state: "state" }, outputs: { toggleExpanded: "toggleExpanded", activate: "activate", deactivate: "deactivate", nodeActivate: "nodeActivate", nodeDeactivate: "nodeDeactivate", select: "select", deselect: "deselect", focus: "focus", blur: "blur", updateData: "updateData", initialized: "initialized", moveNode: "moveNode", copyNode: "copyNode", loadNodeChildren: "loadNodeChildren", changeFilter: "changeFilter", event: "event", stateChange: "stateChange" }, host: { listeners: { "body: keydown": "onKeydown($event)", "body: mousedown": "onMousedown($event)" } }, providers: [TreeModel], queries: [{ propertyName: "loadingTemplate", first: true, predicate: ["loadingTemplate"], descendants: true }, { propertyName: "treeNodeTemplate", first: true, predicate: ["treeNodeTemplate"], descendants: true }, { propertyName: "treeNodeWrapperTemplate", first: true, predicate: ["treeNodeWrapperTemplate"], descendants: true }, { propertyName: "treeNodeFullTemplate", first: true, predicate: ["treeNodeFullTemplate"], descendants: true }], viewQueries: [{ propertyName: "viewportComponent", first: true, predicate: ["viewport"], descendants: true }], usesOnChanges: true, ngImport: i0, template: `
<tree-viewport #viewport>
<div
class="angular-tree-component"
[class.node-dragging]="treeDraggedElement.isDragging()"
[class.angular-tree-component-rtl]="treeModel.options.rtl">
<tree-node-collection
*ngIf="treeModel.roots"
[nodes]="treeModel.roots"
[treeModel]="treeModel"
[templates]="{
loadingTemplate: loadingTemplate,
treeNodeTemplate: treeNodeTemplate,
treeNodeWrapperTemplate: treeNodeWrapperTemplate,
treeNodeFullTemplate: treeNodeFullTemplate
}">
</tree-node-collection>
<tree-node-drop-slot
class="empty-tree-drop-slot"
*ngIf="treeModel.isEmptyTree()"
[dropIndex]="0"
[node]="treeModel.virtualRoot">
</tree-node-drop-slot>
</div>
</tree-viewport>
`, isInline: true, dependencies: [{ kind: "directive", type: i0.forwardRef(function () { return i3.NgIf; }), selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i0.forwardRef(function () { return i4.TreeNodeDropSlot; }), selector: "TreeNodeDropSlot, tree-node-drop-slot", inputs: ["node", "dropIndex"] }, { kind: "component", type: i0.forwardRef(function () { return TreeNodeCollectionComponent; }), selector: "tree-node-collection", inputs: ["nodes", "treeModel", "templates"] }, { kind: "component", type: i0.forwardRef(function () { return i5.TreeViewportComponent; }), selector: "tree-viewport" }] });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeComponent, decorators: [{
type: Component,
args: [{ selector: 'Tree, tree-root', providers: [TreeModel], template: `
<tree-viewport #viewport>
<div
class="angular-tree-component"
[class.node-dragging]="treeDraggedElement.isDragging()"
[class.angular-tree-component-rtl]="treeModel.options.rtl">
<tree-node-collection
*ngIf="treeModel.roots"
[nodes]="treeModel.roots"
[treeModel]="treeModel"
[templates]="{
loadingTemplate: loadingTemplate,
treeNodeTemplate: treeNodeTemplate,
treeNodeWrapperTemplate: treeNodeWrapperTemplate,
treeNodeFullTemplate: treeNodeFullTemplate
}">
</tree-node-collection>
<tree-node-drop-slot
class="empty-tree-drop-slot"
*ngIf="treeModel.isEmptyTree()"
[dropIndex]="0"
[node]="treeModel.virtualRoot">
</tree-node-drop-slot>
</div>
</tree-viewport>
` }]
}], ctorParameters: function () { return [{ type: i1.TreeModel }, { type: i2.TreeDraggedElement }]; }, propDecorators: { loadingTemplate: [{
type: ContentChild,
args: ['loadingTemplate', { static: false }]
}], treeNodeTemplate: [{
type: ContentChild,
args: ['treeNodeTemplate', { static: false }]
}], treeNodeWrapperTemplate: [{
type: ContentChild,
args: ['treeNodeWrapperTemplate', { static: false }]
}], treeNodeFullTemplate: [{
type: ContentChild,
args: ['treeNodeFullTemplate', { static: false }]
}], viewportComponent: [{
type: ViewChild,
args: ['viewport', { static: false }]
}], nodes: [{
type: Input
}], options: [{
type: Input
}], focused: [{
type: Input
}], state: [{
type: Input
}], toggleExpanded: [{
type: Output
}], activate: [{
type: Output
}], deactivate: [{
type: Output
}], nodeActivate: [{
type: Output
}], nodeDeactivate: [{
type: Output
}], select: [{
type: Output
}], deselect: [{
type: Output
}], focus: [{
type: Output
}], blur: [{
type: Output
}], updateData: [{
type: Output
}], initialized: [{
type: Output
}], moveNode: [{
type: Output
}], copyNode: [{
type: Output
}], loadNodeChildren: [{
type: Output
}], changeFilter: [{
type: Output
}], event: [{
type: Output
}], stateChange: [{
type: Output
}], onKeydown: [{
type: HostListener,
args: ['body: keydown', ['$event']]
}], onMousedown: [{
type: HostListener,
args: ['body: mousedown', ['$event']]
}] } });
//imported from tree-node.children to avoid NG3003 cycle error
export class TreeNodeChildrenComponent {
node;
templates;
/** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeNodeChildrenComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
/** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.2", type: TreeNodeChildrenComponent, selector: "tree-node-children", inputs: { node: "node", templates: "templates" }, ngImport: i0, template: `
<ng-container *treeMobxAutorun="{ dontDetach: true }">
<div
[class.tree-children]="true"
[class.tree-children-no-padding]="node.options.levelPadding"
*treeAnimateOpen="
node.isExpanded;
speed: node.options.animateSpeed;
acceleration: node.options.animateAcceleration;
enabled: node.options.animateExpand
"
>
<tree-node-collection
*ngIf="node.children"
[nodes]="node.children"
[templates]="templates"
[treeModel]="node.treeModel"
>
</tree-node-collection>
<tree-loading-component
[style.padding-left]="node.getNodePadding()"
class="tree-node-loading"
*ngIf="!node.children"
[template]="templates.loadingTemplate"
[node]="node"
></tree-loading-component>
</div>
</ng-container>
`, isInline: true, dependencies: [{ kind: "directive", type: i0.forwardRef(function () { return i3.NgIf; }), selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i0.forwardRef(function () { return i6.LoadingComponent; }), selector: "tree-loading-component", inputs: ["template", "node"] }, { kind: "component", type: i0.forwardRef(function () { return TreeNodeCollectionComponent; }), selector: "tree-node-collection", inputs: ["nodes", "treeModel", "templates"] }, { kind: "directive", type: i0.forwardRef(function () { return i7.TreeAnimateOpenDirective; }), selector: "[treeAnimateOpen]", inputs: ["treeAnimateOpenSpeed", "treeAnimateOpenAcceleration", "treeAnimateOpenEnabled", "treeAnimateOpen"] }, { kind: "directive", type: i0.forwardRef(function () { return i8.TreeMobxAutorunDirective; }), selector: "[treeMobxAutorun]", inputs: ["treeMobxAutorun"] }], encapsulation: i0.ViewEncapsulation.None });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeNodeChildrenComponent, decorators: [{
type: Component,
args: [{ selector: 'tree-node-children', encapsulation: ViewEncapsulation.None, template: `
<ng-container *treeMobxAutorun="{ dontDetach: true }">
<div
[class.tree-children]="true"
[class.tree-children-no-padding]="node.options.levelPadding"
*treeAnimateOpen="
node.isExpanded;
speed: node.options.animateSpeed;
acceleration: node.options.animateAcceleration;
enabled: node.options.animateExpand
"
>
<tree-node-collection
*ngIf="node.children"
[nodes]="node.children"
[templates]="templates"
[treeModel]="node.treeModel"
>
</tree-node-collection>
<tree-loading-component
[style.padding-left]="node.getNodePadding()"
class="tree-node-loading"
*ngIf="!node.children"
[template]="templates.loadingTemplate"
[node]="node"
></tree-loading-component>
</div>
</ng-container>
` }]
}], propDecorators: { node: [{
type: Input
}], templates: [{
type: Input
}] } });
//imported from tree-node-collection.component.ts to avoid NG3003 cycle error
export class TreeNodeCollectionComponent {
get nodes() {
return this._nodes;
}
set nodes(nodes) {
this.setNodes(nodes);
}
treeModel;
constructor() {
makeObservable(this, {
_nodes: observable,
viewportNodes: observable,
marginTop: computed,
setNodes: action
});
}
_nodes;
virtualScroll; // Cannot inject this, because we might be inside treeNodeTemplateFull
templates;
viewportNodes;
get marginTop() {
const firstNode = this.viewportNodes && this.viewportNodes.length && this.viewportNodes[0];
const relativePosition = firstNode && firstNode.parent
? firstNode.position -
firstNode.parent.position -
firstNode.parent.getSelfHeight()
: 0;
return `${relativePosition}px`;
}
_dispose = [];
setNodes(nodes) {
this._nodes = nodes;
}
ngOnInit() {
this.virtualScroll = this.treeModel.virtualScroll;
this._dispose = [
// return node indexes so we can compare structurally,
reaction(() => {
return this.virtualScroll
.getViewportNodes(this.nodes)
.map(n => n.index);
}, nodeIndexes => {
this.viewportNodes = nodeIndexes.map(i => this.nodes[i]);
}, { compareStructural: true, fireImmediately: true }),
reaction(() => this.nodes, nodes => {
this.viewportNodes = this.virtualScroll.getViewportNodes(nodes);
})
];
}
ngOnDestroy() {
this._dispose.forEach(d => d());
}
trackNode(index, node) {
return node.id;
}
/** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeNodeCollectionComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
/** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.2", type: TreeNodeCollectionComponent, selector: "tree-node-collection", inputs: { nodes: "nodes", treeModel: "treeModel", templates: "templates" }, ngImport: i0, template: `
<ng-container *treeMobxAutorun="{ dontDetach: true }">
<div [style.margin-top]="marginTop">
<tree-node
*ngFor="let node of viewportNodes; let i = index; trackBy: trackNode"
[node]="node"
[index]="i"
[templates]="templates"
>
</tree-node>
</div>
</ng-container>
`, isInline: true, dependencies: [{ kind: "directive", type: i0.forwardRef(function () { return i3.NgForOf; }), selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i0.forwardRef(function () { return TreeNodeComponent; }), selector: "TreeNode, tree-node", inputs: ["node", "index", "templates"] }, { kind: "directive", type: i0.forwardRef(function () { return i8.TreeMobxAutorunDirective; }), selector: "[treeMobxAutorun]", inputs: ["treeMobxAutorun"] }], encapsulation: i0.ViewEncapsulation.None });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeNodeCollectionComponent, decorators: [{
type: Component,
args: [{
selector: 'tree-node-collection',
encapsulation: ViewEncapsulation.None,
template: `
<ng-container *treeMobxAutorun="{ dontDetach: true }">
<div [style.margin-top]="marginTop">
<tree-node
*ngFor="let node of viewportNodes; let i = index; trackBy: trackNode"
[node]="node"
[index]="i"
[templates]="templates"
>
</tree-node>
</div>
</ng-container>
`
}]
}], ctorParameters: function () { return []; }, propDecorators: { nodes: [{
type: Input
}], treeModel: [{
type: Input
}], templates: [{
type: Input
}] } });
//from tree.node.component.ts to avoid NG3003 cycle error
export class TreeNodeComponent {
node;
index;
templates;
/** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeNodeComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
/** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.2", type: TreeNodeComponent, selector: "TreeNode, tree-node", inputs: { node: "node", index: "index", templates: "templates" }, ngImport: i0, template: `
<ng-container *treeMobxAutorun="{ dontDetach: true }">
<div
*ngIf="!templates.treeNodeFullTemplate"
[class]="node.getClass()"
[class.tree-node]="true"
[class.tree-node-expanded]="node.isExpanded && node.hasChildren"
[class.tree-node-collapsed]="node.isCollapsed && node.hasChildren"
[class.tree-node-leaf]="node.isLeaf"
[class.tree-node-active]="node.isActive"
[class.tree-node-focused]="node.isFocused"
>
<tree-node-drop-slot
*ngIf="index === 0"
[dropIndex]="node.index"
[node]="node.parent"
></tree-node-drop-slot>
<tree-node-wrapper
[node]="node"
[index]="index"
[templates]="templates"
></tree-node-wrapper>
<tree-node-children
[node]="node"
[templates]="templates"
></tree-node-children>
<tree-node-drop-slot
[dropIndex]="node.index + 1"
[node]="node.parent"
></tree-node-drop-slot>
</div>
<ng-container
[ngTemplateOutlet]="templates.treeNodeFullTemplate"
[ngTemplateOutletContext]="{
$implicit: node,
node: node,
index: index,
templates: templates
}"
>
</ng-container>
</ng-container>
`, isInline: true, dependencies: [{ kind: "directive", type: i0.forwardRef(function () { return i3.NgIf; }), selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i0.forwardRef(function () { return i3.NgTemplateOutlet; }), selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i0.forwardRef(function () { return TreeNodeChildrenComponent; }), selector: "tree-node-children", inputs: ["node", "templates"] }, { kind: "component", type: i0.forwardRef(function () { return i4.TreeNodeDropSlot; }), selector: "TreeNodeDropSlot, tree-node-drop-slot", inputs: ["node", "dropIndex"] }, { kind: "component", type: i0.forwardRef(function () { return TreeNodeWrapperComponent; }), selector: "tree-node-wrapper", inputs: ["node", "index", "templates"] }, { kind: "directive", type: i0.forwardRef(function () { return i8.TreeMobxAutorunDirective; }), selector: "[treeMobxAutorun]", inputs: ["treeMobxAutorun"] }], encapsulation: i0.ViewEncapsulation.None });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeNodeComponent, decorators: [{
type: Component,
args: [{ selector: 'TreeNode, tree-node', encapsulation: ViewEncapsulation.None, template: `
<ng-container *treeMobxAutorun="{ dontDetach: true }">
<div
*ngIf="!templates.treeNodeFullTemplate"
[class]="node.getClass()"
[class.tree-node]="true"
[class.tree-node-expanded]="node.isExpanded && node.hasChildren"
[class.tree-node-collapsed]="node.isCollapsed && node.hasChildren"
[class.tree-node-leaf]="node.isLeaf"
[class.tree-node-active]="node.isActive"
[class.tree-node-focused]="node.isFocused"
>
<tree-node-drop-slot
*ngIf="index === 0"
[dropIndex]="node.index"
[node]="node.parent"
></tree-node-drop-slot>
<tree-node-wrapper
[node]="node"
[index]="index"
[templates]="templates"
></tree-node-wrapper>
<tree-node-children
[node]="node"
[templates]="templates"
></tree-node-children>
<tree-node-drop-slot
[dropIndex]="node.index + 1"
[node]="node.parent"
></tree-node-drop-slot>
</div>
<ng-container
[ngTemplateOutlet]="templates.treeNodeFullTemplate"
[ngTemplateOutletContext]="{
$implicit: node,
node: node,
index: index,
templates: templates
}"
>
</ng-container>
</ng-container>
` }]
}], propDecorators: { node: [{
type: Input
}], index: [{
type: Input
}], templates: [{
type: Input
}] } });
//from tree-node-content.component.ts to avoid NG3003 cycle error.
export class TreeNodeContent {
node;
index;
template;
/** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeNodeContent, deps: [], target: i0.ɵɵFactoryTarget.Component });
/** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.2", type: TreeNodeContent, selector: "tree-node-content", inputs: { node: "node", index: "index", template: "template" }, ngImport: i0, template: `
<span *ngIf="!template">{{ node.displayField }}</span>
<ng-container
[ngTemplateOutlet]="template"
[ngTemplateOutletContext]="{ $implicit: node, node: node, index: index }">
</ng-container>`, isInline: true, dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], encapsulation: i0.ViewEncapsulation.None });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeNodeContent, decorators: [{
type: Component,
args: [{
selector: 'tree-node-content',
encapsulation: ViewEncapsulation.None,
template: `
<span *ngIf="!template">{{ node.displayField }}</span>
<ng-container
[ngTemplateOutlet]="template"
[ngTemplateOutletContext]="{ $implicit: node, node: node, index: index }">
</ng-container>`,
}]
}], propDecorators: { node: [{
type: Input
}], index: [{
type: Input
}], template: [{
type: Input
}] } });
//from tree-node-wrapper.component.ts file to avoid NG3003 cycle error.
export class TreeNodeWrapperComponent {
node;
index;
templates;
/** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeNodeWrapperComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
/** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.2", type: TreeNodeWrapperComponent, selector: "tree-node-wrapper", inputs: { node: "node", index: "index", templates: "templates" }, ngImport: i0, template: `
<div *ngIf="!templates.treeNodeWrapperTemplate" class="node-wrapper" [style.padding-left]="node.getNodePadding()">
<tree-node-checkbox *ngIf="node.options.useCheckbox" [node]="node"></tree-node-checkbox>
<tree-node-expander [node]="node"></tree-node-expander>
<div class="node-content-wrapper"
[class.node-content-wrapper-active]="node.isActive"
[class.node-content-wrapper-focused]="node.isFocused"
(click)="node.mouseAction('click', $event)"
(dblclick)="node.mouseAction('dblClick', $event)"
(mouseover)="node.mouseAction('mouseOver', $event)"
(mouseout)="node.mouseAction('mouseOut', $event)"
(contextmenu)="node.mouseAction('contextMenu', $event)"
(treeDrop)="node.onDrop($event)"
(treeDropDragOver)="node.mouseAction('dragOver', $event)"
(treeDropDragLeave)="node.mouseAction('dragLeave', $event)"
(treeDropDragEnter)="node.mouseAction('dragEnter', $event)"
[treeAllowDrop]="node.allowDrop"
[allowDragoverStyling]="node.allowDragoverStyling()"
[treeDrag]="node"
[treeDragEnabled]="node.allowDrag()">
<tree-node-content [node]="node" [index]="index" [template]="templates.treeNodeTemplate">
</tree-node-content>
</div>
</div>
<ng-container
[ngTemplateOutlet]="templates.treeNodeWrapperTemplate"
[ngTemplateOutletContext]="{ $implicit: node, node: node, index: index, templates: templates }">
</ng-container>
`, isInline: true, dependencies: [{ kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: TreeNodeContent, selector: "tree-node-content", inputs: ["node", "index", "template"] }, { kind: "directive", type: i9.TreeDropDirective, selector: "[treeDrop]", inputs: ["allowDragoverStyling", "treeAllowDrop"], outputs: ["treeDrop", "treeDropDragOver", "treeDropDragLeave", "treeDropDragEnter"] }, { kind: "directive", type: i10.TreeDragDirective, selector: "[treeDrag]", inputs: ["treeDrag", "treeDragEnabled"] }, { kind: "component", type: i11.TreeNodeExpanderComponent, selector: "tree-node-expander", inputs: ["node"] }, { kind: "component", type: i12.TreeNodeCheckboxComponent, selector: "tree-node-checkbox", inputs: ["node"] }], encapsulation: i0.ViewEncapsulation.None });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: TreeNodeWrapperComponent, decorators: [{
type: Component,
args: [{ selector: 'tree-node-wrapper', encapsulation: ViewEncapsulation.None, template: `
<div *ngIf="!templates.treeNodeWrapperTemplate" class="node-wrapper" [style.padding-left]="node.getNodePadding()">
<tree-node-checkbox *ngIf="node.options.useCheckbox" [node]="node"></tree-node-checkbox>
<tree-node-expander [node]="node"></tree-node-expander>
<div class="node-content-wrapper"
[class.node-content-wrapper-active]="node.isActive"
[class.node-content-wrapper-focused]="node.isFocused"
(click)="node.mouseAction('click', $event)"
(dblclick)="node.mouseAction('dblClick', $event)"
(mouseover)="node.mouseAction('mouseOver', $event)"
(mouseout)="node.mouseAction('mouseOut', $event)"
(contextmenu)="node.mouseAction('contextMenu', $event)"
(treeDrop)="node.onDrop($event)"
(treeDropDragOver)="node.mouseAction('dragOver', $event)"
(treeDropDragLeave)="node.mouseAction('dragLeave', $event)"
(treeDropDragEnter)="node.mouseAction('dragEnter', $event)"
[treeAllowDrop]="node.allowDrop"
[allowDragoverStyling]="node.allowDragoverStyling()"
[treeDrag]="node"
[treeDragEnabled]="node.allowDrag()">
<tree-node-content [node]="node" [index]="index" [template]="templates.treeNodeTemplate">
</tree-node-content>
</div>
</div>
<ng-container
[ngTemplateOutlet]="templates.treeNodeWrapperTemplate"
[ngTemplateOutletContext]="{ $implicit: node, node: node, index: index, templates: templates }">
</ng-container>
` }]
}], propDecorators: { node: [{
type: Input
}], index: [{
type: Input
}], templates: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree.component.js","sourceRoot":"","sources":["../../../../../projects/angular-tree-component/src/lib/components/tree.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, ContentChild, EventEmitter, HostListener, Input, OnChanges, Output, TemplateRef, ViewChild, ViewEncapsulation, OnInit, OnDestroy } from '@angular/core';\r\nimport { TreeModel } from '../models/tree.model';\r\nimport { TreeDraggedElement } from '../models/tree-dragged-element.model';\r\nimport { TreeOptions } from '../models/tree-options.model';\r\nimport { ITreeOptions } from '../defs/api';\r\nimport { TreeViewportComponent } from './tree-viewport.component';\r\n\r\nimport { TreeNode } from '../models/tree-node.model';\r\n\r\nimport { makeObservable, reaction } from 'mobx';\r\nimport { observable, computed, action } from '../mobx-angular/mobx-proxy';\r\nimport { TreeVirtualScroll } from '../models/tree-virtual-scroll.model';\r\n\r\n@Component({\r\n  selector: 'Tree, tree-root',\r\n  providers: [TreeModel],\r\n  styles: [],\r\n  template: `\r\n      <tree-viewport #viewport>\r\n          <div\r\n                  class=\"angular-tree-component\"\r\n                  [class.node-dragging]=\"treeDraggedElement.isDragging()\"\r\n                  [class.angular-tree-component-rtl]=\"treeModel.options.rtl\">\r\n              <tree-node-collection\r\n                      *ngIf=\"treeModel.roots\"\r\n                      [nodes]=\"treeModel.roots\"\r\n                      [treeModel]=\"treeModel\"\r\n                      [templates]=\"{\r\n            loadingTemplate: loadingTemplate,\r\n            treeNodeTemplate: treeNodeTemplate,\r\n            treeNodeWrapperTemplate: treeNodeWrapperTemplate,\r\n            treeNodeFullTemplate: treeNodeFullTemplate\r\n          }\">\r\n              </tree-node-collection>\r\n              <tree-node-drop-slot\r\n                      class=\"empty-tree-drop-slot\"\r\n                      *ngIf=\"treeModel.isEmptyTree()\"\r\n                      [dropIndex]=\"0\"\r\n                      [node]=\"treeModel.virtualRoot\">\r\n              </tree-node-drop-slot>\r\n          </div>\r\n      </tree-viewport>\r\n  `\r\n})\r\nexport class TreeComponent implements OnChanges {\r\n  _nodes: any[];\r\n  _options: TreeOptions;\r\n\r\n  @ContentChild('loadingTemplate', { static: false }) loadingTemplate: TemplateRef<any>;\r\n  @ContentChild('treeNodeTemplate', { static: false }) treeNodeTemplate: TemplateRef<any>;\r\n  @ContentChild('treeNodeWrapperTemplate', { static: false }) treeNodeWrapperTemplate: TemplateRef<any>;\r\n  @ContentChild('treeNodeFullTemplate', { static: false }) treeNodeFullTemplate: TemplateRef<any>;\r\n  @ViewChild('viewport', { static: false }) viewportComponent: TreeViewportComponent;\r\n\r\n  // Will be handled in ngOnChanges\r\n  @Input() set nodes(nodes: any[]) {\r\n  };\r\n\r\n  @Input() set options(options: ITreeOptions) {\r\n  };\r\n\r\n  @Input() set focused(value: boolean) {\r\n    this.treeModel.setFocus(value);\r\n  }\r\n\r\n  @Input() set state(state) {\r\n    this.treeModel.setState(state);\r\n  }\r\n\r\n  @Output() toggleExpanded;\r\n  @Output() activate;\r\n  @Output() deactivate;\r\n  @Output() nodeActivate;\r\n  @Output() nodeDeactivate;\r\n  @Output() select;\r\n  @Output() deselect;\r\n  @Output() focus;\r\n  @Output() blur;\r\n  @Output() updateData;\r\n  @Output() initialized;\r\n  @Output() moveNode;\r\n  @Output() copyNode;\r\n  @Output() loadNodeChildren;\r\n  @Output() changeFilter;\r\n  @Output() event;\r\n  @Output() stateChange;\r\n\r\n  constructor(\r\n    public treeModel: TreeModel,\r\n    public treeDraggedElement: TreeDraggedElement) {\r\n\r\n    treeModel.eventNames.forEach((name) => this[name] = new EventEmitter());\r\n    treeModel.subscribeToState((state) => this.stateChange.emit(state));\r\n  }\r\n\r\n  @HostListener('body: keydown', ['$event'])\r\n  onKeydown($event) {\r\n    if (!this.treeModel.isFocused) return;\r\n    if (['input', 'textarea'].includes(document.activeElement.tagName.toLowerCase())) return;\r\n\r\n    const focusedNode = this.treeModel.getFocusedNode();\r\n\r\n    this.treeModel.performKeyAction(focusedNode, $event);\r\n  }\r\n\r\n  @HostListener('body: mousedown', ['$event'])\r\n  onMousedown($event) {\r\n    function isOutsideClick(startElement: Element, nodeName: string) {\r\n      return !startElement ? true : startElement.localName === nodeName ? false : isOutsideClick(startElement.parentElement, nodeName);\r\n    }\r\n\r\n    if (isOutsideClick($event.target, 'tree-root')) {\r\n      this.treeModel.setFocus(false);\r\n    }\r\n  }\r\n\r\n  ngOnChanges(changes) {\r\n    if (changes.options || changes.nodes) {\r\n      this.treeModel.setData({\r\n        options: changes.options && changes.options.currentValue,\r\n        nodes: changes.nodes && changes.nodes.currentValue,\r\n        events: this.pick(this, this.treeModel.eventNames)\r\n      });\r\n    }\r\n  }\r\n\r\n  sizeChanged() {\r\n    this.viewportComponent.setViewport();\r\n  }\r\n\r\n  private pick(object, keys) {\r\n    return keys.reduce((obj, key) => {\r\n      if (object && object.hasOwnProperty(key)) {\r\n        obj[key] = object[key];\r\n      }\r\n      return obj;\r\n    }, {});\r\n  }\r\n}\r\n\r\n//imported from tree-node.children to avoid NG3003 cycle error\r\n\r\n@Component({\r\n  selector: 'tree-node-children',\r\n  encapsulation: ViewEncapsulation.None,\r\n  styles: [],\r\n  template: `\r\n    <ng-container *treeMobxAutorun=\"{ dontDetach: true }\">\r\n      <div\r\n        [class.tree-children]=\"true\"\r\n        [class.tree-children-no-padding]=\"node.options.levelPadding\"\r\n        *treeAnimateOpen=\"\r\n          node.isExpanded;\r\n          speed: node.options.animateSpeed;\r\n          acceleration: node.options.animateAcceleration;\r\n          enabled: node.options.animateExpand\r\n        \"\r\n      >\r\n        <tree-node-collection\r\n          *ngIf=\"node.children\"\r\n          [nodes]=\"node.children\"\r\n          [templates]=\"templates\"\r\n          [treeModel]=\"node.treeModel\"\r\n        >\r\n        </tree-node-collection>\r\n        <tree-loading-component\r\n          [style.padding-left]=\"node.getNodePadding()\"\r\n          class=\"tree-node-loading\"\r\n          *ngIf=\"!node.children\"\r\n          [template]=\"templates.loadingTemplate\"\r\n          [node]=\"node\"\r\n        ></tree-loading-component>\r\n      </div>\r\n    </ng-container>\r\n  `\r\n})\r\nexport class TreeNodeChildrenComponent {\r\n  @Input() node: TreeNode;\r\n  @Input() templates: any;\r\n}\r\n\r\n\r\n//imported from tree-node-collection.component.ts to avoid NG3003 cycle error\r\n\r\n@Component({\r\n  selector: 'tree-node-collection',\r\n  encapsulation: ViewEncapsulation.None,\r\n  template: `\r\n    <ng-container *treeMobxAutorun=\"{ dontDetach: true }\">\r\n      <div [style.margin-top]=\"marginTop\">\r\n        <tree-node\r\n          *ngFor=\"let node of viewportNodes; let i = index; trackBy: trackNode\"\r\n          [node]=\"node\"\r\n          [index]=\"i\"\r\n          [templates]=\"templates\"\r\n        >\r\n        </tree-node>\r\n      </div>\r\n    </ng-container>\r\n  `\r\n})\r\nexport class TreeNodeCollectionComponent implements OnInit, OnDestroy {\r\n  @Input()\r\n  get nodes() {\r\n    return this._nodes;\r\n  }\r\n  se