ng2-tree
Version:
angular2 component for visualizing data that can be naturally represented as a tree
399 lines (383 loc) • 49.8 kB
JavaScript
import { Component, Input, ViewChild } from '@angular/core';
import { Ng2TreeSettings } from './tree.types';
import { Tree } from './tree';
import { TreeController } from './tree-controller';
import { NodeMenuItemAction } from './menu/menu.events';
import { NodeEditableEventAction } from './editable/editable.events';
import * as EventUtils from './utils/event.utils';
import { get, isNil } from './utils/fn.utils';
import { merge } from 'rxjs';
import { filter } from 'rxjs/operators';
import * as i0 from "@angular/core";
import * as i1 from "./menu/node-menu.service";
import * as i2 from "./tree.service";
import * as i3 from "@angular/common";
import * as i4 from "./draggable/node-draggable.directive";
import * as i5 from "./editable/node-editable.directive";
import * as i6 from "./menu/node-menu.component";
import * as i7 from "./utils/safe-html.pipe";
export class TreeInternalComponent {
nodeMenuService;
treeService;
nodeElementRef;
tree;
settings;
template;
isSelected = false;
isRightMenuVisible = false;
isLeftMenuVisible = false;
isReadOnly = false;
controller;
checkboxElementRef;
subscriptions = [];
constructor(nodeMenuService, treeService, nodeElementRef) {
this.nodeMenuService = nodeMenuService;
this.treeService = treeService;
this.nodeElementRef = nodeElementRef;
}
ngAfterViewInit() {
if (this.tree.checked && !this.tree.firstCheckedFired) {
this.tree.firstCheckedFired = true;
this.treeService.fireNodeChecked(this.tree);
}
}
ngOnInit() {
const nodeId = get(this.tree, 'node.id', '');
if (nodeId) {
this.controller = new TreeController(this);
this.treeService.setController(nodeId, this.controller);
}
this.settings = this.settings || new Ng2TreeSettings();
this.isReadOnly = !get(this.settings, 'enableCheckboxes', true);
if (this.tree.isRoot() && this.settings.rootIsVisible === false) {
this.tree.disableCollapseOnInit();
}
this.subscriptions.push(this.nodeMenuService.hideMenuStream(this.nodeElementRef).subscribe(() => {
this.isRightMenuVisible = false;
this.isLeftMenuVisible = false;
}));
this.subscriptions.push(this.treeService.unselectStream(this.tree).subscribe(() => (this.isSelected = false)));
this.subscriptions.push(this.treeService.draggedStream(this.tree, this.nodeElementRef).subscribe((e) => {
if (this.tree.hasSibling(e.captured.tree)) {
this.swapWithSibling(e.captured.tree, this.tree);
}
else if (this.tree.isBranch()) {
this.moveNodeToThisTreeAndRemoveFromPreviousOne(e, this.tree);
}
else {
this.moveNodeToParentTreeAndRemoveFromPreviousOne(e, this.tree);
}
}));
this.subscriptions.push(merge(this.treeService.nodeChecked$, this.treeService.nodeUnchecked$)
.pipe(filter((e) => this.eventContainsId(e) && this.tree.hasChild(e.node)))
.subscribe((e) => this.updateCheckboxState()));
}
ngOnChanges(changes) {
this.controller = new TreeController(this);
}
ngOnDestroy() {
if (get(this.tree, 'node.id', '')) {
this.treeService.deleteController(this.tree.node.id);
}
this.subscriptions.forEach(sub => sub && sub.unsubscribe());
}
swapWithSibling(sibling, tree) {
tree.swapWithSibling(sibling);
this.treeService.fireNodeMoved(sibling, sibling.parent);
}
moveNodeToThisTreeAndRemoveFromPreviousOne(e, tree) {
this.treeService.fireNodeRemoved(e.captured.tree);
const addedChild = tree.addChild(e.captured.tree);
this.treeService.fireNodeMoved(addedChild, e.captured.tree.parent);
}
moveNodeToParentTreeAndRemoveFromPreviousOne(e, tree) {
this.treeService.fireNodeRemoved(e.captured.tree);
const addedSibling = tree.addSibling(e.captured.tree, tree.positionInParent);
this.treeService.fireNodeMoved(addedSibling, e.captured.tree.parent);
}
onNodeSelected(e) {
if (!this.tree.selectionAllowed) {
return;
}
if (EventUtils.isLeftButtonClicked(e)) {
this.isSelected = true;
this.treeService.fireNodeSelected(this.tree);
}
}
onNodeUnselected(e) {
if (!this.tree.selectionAllowed) {
return;
}
if (EventUtils.isLeftButtonClicked(e)) {
this.isSelected = false;
this.treeService.fireNodeUnselected(this.tree);
}
}
showRightMenu(e) {
if (!this.tree.hasRightMenu()) {
return;
}
if (EventUtils.isRightButtonClicked(e)) {
this.isRightMenuVisible = !this.isRightMenuVisible;
this.nodeMenuService.hideMenuForAllNodesExcept(this.nodeElementRef);
}
e.preventDefault();
}
showLeftMenu(e) {
if (!this.tree.hasLeftMenu()) {
return;
}
if (EventUtils.isLeftButtonClicked(e)) {
this.isLeftMenuVisible = !this.isLeftMenuVisible;
this.nodeMenuService.hideMenuForAllNodesExcept(this.nodeElementRef);
if (this.isLeftMenuVisible) {
e.preventDefault();
}
}
}
onMenuItemSelected(e) {
switch (e.nodeMenuItemAction) {
case NodeMenuItemAction.NewTag:
this.onNewSelected(e);
break;
case NodeMenuItemAction.NewFolder:
this.onNewSelected(e);
break;
case NodeMenuItemAction.Rename:
this.onRenameSelected();
break;
case NodeMenuItemAction.Remove:
this.onRemoveSelected();
break;
case NodeMenuItemAction.Custom:
this.onCustomSelected();
this.treeService.fireMenuItemSelected(this.tree, e.nodeMenuItemSelected);
break;
default:
throw new Error(`Chosen menu item doesn't exist`);
}
}
onNewSelected(e) {
this.tree.createNode(e.nodeMenuItemAction === NodeMenuItemAction.NewFolder);
this.isRightMenuVisible = false;
this.isLeftMenuVisible = false;
}
onRenameSelected() {
this.tree.markAsBeingRenamed();
this.isRightMenuVisible = false;
this.isLeftMenuVisible = false;
}
onRemoveSelected() {
this.treeService.deleteController(get(this.tree, 'node.id', ''));
this.treeService.fireNodeRemoved(this.tree);
}
onCustomSelected() {
this.isRightMenuVisible = false;
this.isLeftMenuVisible = false;
}
onSwitchFoldingType() {
this.tree.switchFoldingType();
this.treeService.fireNodeSwitchFoldingType(this.tree);
}
applyNewValue(e) {
if ((e.action === NodeEditableEventAction.Cancel || this.tree.isNew()) && Tree.isValueEmpty(e.value)) {
return this.treeService.fireNodeRemoved(this.tree);
}
if (this.tree.isNew()) {
this.tree.value = e.value;
this.treeService.fireNodeCreated(this.tree);
}
if (this.tree.isBeingRenamed()) {
const oldValue = this.tree.value;
this.tree.value = e.value;
this.treeService.fireNodeRenamed(oldValue, this.tree);
}
this.tree.markAsModified();
}
shouldShowInputForTreeValue() {
return this.tree.isNew() || this.tree.isBeingRenamed();
}
isRootHidden() {
return this.tree.isRoot() && !this.settings.rootIsVisible;
}
hasCustomMenu() {
return this.tree.hasCustomMenu();
}
switchNodeCheckStatus() {
if (!this.tree.checked) {
this.onNodeChecked();
}
else {
this.onNodeUnchecked();
}
}
onNodeChecked() {
if (!this.checkboxElementRef) {
return;
}
this.checkboxElementRef.nativeElement.indeterminate = false;
this.treeService.fireNodeChecked(this.tree);
this.executeOnChildController(controller => controller.check());
this.tree.checked = true;
}
onNodeUnchecked() {
if (!this.checkboxElementRef) {
return;
}
this.checkboxElementRef.nativeElement.indeterminate = false;
this.treeService.fireNodeUnchecked(this.tree);
this.executeOnChildController(controller => controller.uncheck());
this.tree.checked = false;
}
executeOnChildController(executor) {
if (this.tree.hasLoadedChildern()) {
this.tree.children.forEach((child) => {
const controller = this.treeService.getController(child.id);
if (!isNil(controller)) {
executor(controller);
}
});
}
}
updateCheckboxState() {
// Calling setTimeout so the value of isChecked will be updated and after that I'll check the children status.
setTimeout(() => {
const checkedChildrenAmount = this.tree.checkedChildrenAmount();
if (checkedChildrenAmount === 0) {
this.checkboxElementRef.nativeElement.indeterminate = false;
this.tree.checked = false;
this.treeService.fireNodeUnchecked(this.tree);
}
else if (checkedChildrenAmount === this.tree.loadedChildrenAmount()) {
this.checkboxElementRef.nativeElement.indeterminate = false;
this.tree.checked = true;
this.treeService.fireNodeChecked(this.tree);
}
else {
this.tree.checked = false;
this.checkboxElementRef.nativeElement.indeterminate = true;
this.treeService.fireNodeIndetermined(this.tree);
}
});
}
eventContainsId(event) {
if (!event.node.id) {
console.warn('"Node with checkbox" feature requires a unique id assigned to every node, please consider to add it.');
return false;
}
return true;
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TreeInternalComponent, deps: [{ token: i1.NodeMenuService }, { token: i2.TreeService }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: TreeInternalComponent, selector: "tree-internal", inputs: { tree: "tree", settings: "settings", template: "template" }, viewQueries: [{ propertyName: "checkboxElementRef", first: true, predicate: ["checkbox"], descendants: true }], usesOnChanges: true, ngImport: i0, template: `
<ul class="tree" *ngIf="tree" [ngClass]="{rootless: isRootHidden()}">
<li>
<div class="value-container"
[ngClass]="{rootless: isRootHidden()}"
[class.selected]="isSelected"
(contextmenu)="showRightMenu($event)"
[nodeDraggable]="nodeElementRef"
[tree]="tree">
<div class="folding" (click)="onSwitchFoldingType()" [ngClass]="tree.foldingCssClass"></div>
<div class="node-checkbox" *ngIf="settings.showCheckboxes">
<input checkbox type="checkbox" [disabled]="isReadOnly" [checked]="this.tree.checked" (change)="switchNodeCheckStatus()" #checkbox />
</div>
<div class="node-value"
*ngIf="!shouldShowInputForTreeValue()"
[class.node-selected]="isSelected"
(click)="onNodeSelected($event)">
<div *ngIf="tree.nodeTemplate" class="node-template" [innerHTML]="tree.nodeTemplate | safeHtml"></div>
<span *ngIf="!template" class="node-name" [innerHTML]="tree.value | safeHtml"></span>
<span class="loading-children" *ngIf="tree.childrenAreBeingLoaded()"></span>
<ng-template [ngTemplateOutlet]="template" [ngTemplateOutletContext]="{ $implicit: tree.node }"></ng-template>
</div>
<input type="text" class="node-value"
*ngIf="shouldShowInputForTreeValue()"
[nodeEditable]="tree.value"
(valueChanged)="applyNewValue($event)"/>
<div class="node-left-menu" *ngIf="tree.hasLeftMenu()" (click)="showLeftMenu($event)" [innerHTML]="tree.leftMenuTemplate">
</div>
<node-menu *ngIf="tree.hasLeftMenu() && isLeftMenuVisible && !hasCustomMenu()"
(menuItemSelected)="onMenuItemSelected($event)">
</node-menu>
</div>
<node-menu *ngIf="isRightMenuVisible && !hasCustomMenu()"
(menuItemSelected)="onMenuItemSelected($event)">
</node-menu>
<node-menu *ngIf="hasCustomMenu() && (isRightMenuVisible || isLeftMenuVisible)"
[menuItems]="tree.menuItems"
(menuItemSelected)="onMenuItemSelected($event)">
</node-menu>
<div *ngIf="tree.keepNodesInDOM()" [ngStyle]="{'display': tree.isNodeExpanded() ? 'block' : 'none'}">
<tree-internal *ngFor="let child of tree.childrenAsync | async" [tree]="child" [template]="template" [settings]="settings"></tree-internal>
</div>
<ng-template [ngIf]="tree.isNodeExpanded() && !tree.keepNodesInDOM()">
<tree-internal *ngFor="let child of tree.childrenAsync | async" [tree]="child" [template]="template" [settings]="settings"></tree-internal>
</ng-template>
</li>
</ul>
`, isInline: true, dependencies: [{ kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i4.NodeDraggableDirective, selector: "[nodeDraggable]", inputs: ["nodeDraggable", "tree"] }, { kind: "directive", type: i5.NodeEditableDirective, selector: "[nodeEditable]", inputs: ["nodeEditable"], outputs: ["valueChanged"] }, { kind: "component", type: i6.NodeMenuComponent, selector: "node-menu", inputs: ["menuItems"], outputs: ["menuItemSelected"] }, { kind: "component", type: TreeInternalComponent, selector: "tree-internal", inputs: ["tree", "settings", "template"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }, { kind: "pipe", type: i7.SafeHtmlPipe, name: "safeHtml" }] });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TreeInternalComponent, decorators: [{
type: Component,
args: [{
selector: 'tree-internal',
template: `
<ul class="tree" *ngIf="tree" [ngClass]="{rootless: isRootHidden()}">
<li>
<div class="value-container"
[ngClass]="{rootless: isRootHidden()}"
[class.selected]="isSelected"
(contextmenu)="showRightMenu($event)"
[nodeDraggable]="nodeElementRef"
[tree]="tree">
<div class="folding" (click)="onSwitchFoldingType()" [ngClass]="tree.foldingCssClass"></div>
<div class="node-checkbox" *ngIf="settings.showCheckboxes">
<input checkbox type="checkbox" [disabled]="isReadOnly" [checked]="this.tree.checked" (change)="switchNodeCheckStatus()" #checkbox />
</div>
<div class="node-value"
*ngIf="!shouldShowInputForTreeValue()"
[class.node-selected]="isSelected"
(click)="onNodeSelected($event)">
<div *ngIf="tree.nodeTemplate" class="node-template" [innerHTML]="tree.nodeTemplate | safeHtml"></div>
<span *ngIf="!template" class="node-name" [innerHTML]="tree.value | safeHtml"></span>
<span class="loading-children" *ngIf="tree.childrenAreBeingLoaded()"></span>
<ng-template [ngTemplateOutlet]="template" [ngTemplateOutletContext]="{ $implicit: tree.node }"></ng-template>
</div>
<input type="text" class="node-value"
*ngIf="shouldShowInputForTreeValue()"
[nodeEditable]="tree.value"
(valueChanged)="applyNewValue($event)"/>
<div class="node-left-menu" *ngIf="tree.hasLeftMenu()" (click)="showLeftMenu($event)" [innerHTML]="tree.leftMenuTemplate">
</div>
<node-menu *ngIf="tree.hasLeftMenu() && isLeftMenuVisible && !hasCustomMenu()"
(menuItemSelected)="onMenuItemSelected($event)">
</node-menu>
</div>
<node-menu *ngIf="isRightMenuVisible && !hasCustomMenu()"
(menuItemSelected)="onMenuItemSelected($event)">
</node-menu>
<node-menu *ngIf="hasCustomMenu() && (isRightMenuVisible || isLeftMenuVisible)"
[menuItems]="tree.menuItems"
(menuItemSelected)="onMenuItemSelected($event)">
</node-menu>
<div *ngIf="tree.keepNodesInDOM()" [ngStyle]="{'display': tree.isNodeExpanded() ? 'block' : 'none'}">
<tree-internal *ngFor="let child of tree.childrenAsync | async" [tree]="child" [template]="template" [settings]="settings"></tree-internal>
</div>
<ng-template [ngIf]="tree.isNodeExpanded() && !tree.keepNodesInDOM()">
<tree-internal *ngFor="let child of tree.childrenAsync | async" [tree]="child" [template]="template" [settings]="settings"></tree-internal>
</ng-template>
</li>
</ul>
`
}]
}], ctorParameters: function () { return [{ type: i1.NodeMenuService }, { type: i2.TreeService }, { type: i0.ElementRef }]; }, propDecorators: { tree: [{
type: Input
}], settings: [{
type: Input
}], template: [{
type: Input
}], checkboxElementRef: [{
type: ViewChild,
args: ['checkbox', { static: false }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree-internal.component.js","sourceRoot":"","sources":["../../../src/tree-internal.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  ElementRef,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  SimpleChanges,\n  TemplateRef,\n  ViewChild\n} from '@angular/core';\n\nimport * as TreeTypes from './tree.types';\nimport { Ng2TreeSettings } from './tree.types';\nimport { Tree } from './tree';\nimport { TreeController } from './tree-controller';\nimport { NodeMenuService } from './menu/node-menu.service';\nimport { NodeMenuItemAction, NodeMenuItemSelectedEvent } from './menu/menu.events';\nimport { NodeEditableEvent, NodeEditableEventAction } from './editable/editable.events';\nimport { NodeCheckedEvent, NodeEvent } from './tree.events';\nimport { TreeService } from './tree.service';\nimport * as EventUtils from './utils/event.utils';\nimport { NodeDraggableEvent } from './draggable/draggable.events';\nimport { get, isNil } from './utils/fn.utils';\nimport { Subscription } from 'rxjs';\nimport { merge, of } from 'rxjs';\nimport { filter } from 'rxjs/operators';\n\n@Component({\n  selector: 'tree-internal',\n  template: `\n  <ul class=\"tree\" *ngIf=\"tree\" [ngClass]=\"{rootless: isRootHidden()}\">\n    <li>\n      <div class=\"value-container\"\n        [ngClass]=\"{rootless: isRootHidden()}\"\n        [class.selected]=\"isSelected\"\n        (contextmenu)=\"showRightMenu($event)\"\n        [nodeDraggable]=\"nodeElementRef\"\n        [tree]=\"tree\">\n\n        <div class=\"folding\" (click)=\"onSwitchFoldingType()\" [ngClass]=\"tree.foldingCssClass\"></div>\n\n        <div class=\"node-checkbox\" *ngIf=\"settings.showCheckboxes\">\n        <input checkbox  type=\"checkbox\" [disabled]=\"isReadOnly\" [checked]=\"this.tree.checked\" (change)=\"switchNodeCheckStatus()\" #checkbox />\n         </div>\n\n        <div class=\"node-value\"\n          *ngIf=\"!shouldShowInputForTreeValue()\"\n          [class.node-selected]=\"isSelected\"\n          (click)=\"onNodeSelected($event)\">\n            <div *ngIf=\"tree.nodeTemplate\" class=\"node-template\" [innerHTML]=\"tree.nodeTemplate | safeHtml\"></div>\n            <span *ngIf=\"!template\" class=\"node-name\" [innerHTML]=\"tree.value | safeHtml\"></span>\n            <span class=\"loading-children\" *ngIf=\"tree.childrenAreBeingLoaded()\"></span>\n            <ng-template [ngTemplateOutlet]=\"template\" [ngTemplateOutletContext]=\"{ $implicit: tree.node }\"></ng-template>\n        </div>\n\n        <input type=\"text\" class=\"node-value\"\n           *ngIf=\"shouldShowInputForTreeValue()\"\n           [nodeEditable]=\"tree.value\"\n           (valueChanged)=\"applyNewValue($event)\"/>\n\n        <div class=\"node-left-menu\" *ngIf=\"tree.hasLeftMenu()\" (click)=\"showLeftMenu($event)\" [innerHTML]=\"tree.leftMenuTemplate\">\n        </div>\n        <node-menu *ngIf=\"tree.hasLeftMenu() && isLeftMenuVisible && !hasCustomMenu()\"\n          (menuItemSelected)=\"onMenuItemSelected($event)\">\n        </node-menu>\n      </div>\n\n      <node-menu *ngIf=\"isRightMenuVisible && !hasCustomMenu()\"\n           (menuItemSelected)=\"onMenuItemSelected($event)\">\n      </node-menu>\n\n      <node-menu *ngIf=\"hasCustomMenu() && (isRightMenuVisible || isLeftMenuVisible)\"\n           [menuItems]=\"tree.menuItems\"\n           (menuItemSelected)=\"onMenuItemSelected($event)\">\n      </node-menu>\n\n      <div *ngIf=\"tree.keepNodesInDOM()\" [ngStyle]=\"{'display': tree.isNodeExpanded() ? 'block' : 'none'}\">\n        <tree-internal *ngFor=\"let child of tree.childrenAsync | async\" [tree]=\"child\" [template]=\"template\" [settings]=\"settings\"></tree-internal>\n      </div>\n      <ng-template [ngIf]=\"tree.isNodeExpanded() && !tree.keepNodesInDOM()\">\n        <tree-internal *ngFor=\"let child of tree.childrenAsync | async\" [tree]=\"child\" [template]=\"template\" [settings]=\"settings\"></tree-internal>\n      </ng-template>\n    </li>\n  </ul>\n  `\n})\nexport class TreeInternalComponent implements OnInit, OnChanges, OnDestroy, AfterViewInit {\n  @Input() public tree: Tree;\n\n  @Input() public settings: TreeTypes.Ng2TreeSettings;\n\n  @Input() public template: TemplateRef<any>;\n\n  public isSelected = false;\n  public isRightMenuVisible = false;\n  public isLeftMenuVisible = false;\n  public isReadOnly = false;\n  public controller: TreeController;\n\n  @ViewChild('checkbox', { static: false })\n  public checkboxElementRef: ElementRef;\n\n  private subscriptions: Subscription[] = [];\n\n  public constructor(\n    private nodeMenuService: NodeMenuService,\n    public treeService: TreeService,\n    public nodeElementRef: ElementRef\n  ) {}\n\n  public ngAfterViewInit(): void {\n    if (this.tree.checked && !(this.tree as any).firstCheckedFired) {\n      (this.tree as any).firstCheckedFired = true;\n      this.treeService.fireNodeChecked(this.tree);\n    }\n  }\n\n  public ngOnInit(): void {\n    const nodeId = get(this.tree, 'node.id', '');\n    if (nodeId) {\n      this.controller = new TreeController(this);\n      this.treeService.setController(nodeId, this.controller);\n    }\n\n    this.settings = this.settings || new Ng2TreeSettings();\n    this.isReadOnly = !get(this.settings, 'enableCheckboxes', true);\n\n    if (this.tree.isRoot() && this.settings.rootIsVisible === false) {\n      this.tree.disableCollapseOnInit();\n    }\n\n    this.subscriptions.push(\n      this.nodeMenuService.hideMenuStream(this.nodeElementRef).subscribe(() => {\n        this.isRightMenuVisible = false;\n        this.isLeftMenuVisible = false;\n      })\n    );\n\n    this.subscriptions.push(this.treeService.unselectStream(this.tree).subscribe(() => (this.isSelected = false)));\n\n    this.subscriptions.push(\n      this.treeService.draggedStream(this.tree, this.nodeElementRef).subscribe((e: NodeDraggableEvent) => {\n        if (this.tree.hasSibling(e.captured.tree)) {\n          this.swapWithSibling(e.captured.tree, this.tree);\n        } else if (this.tree.isBranch()) {\n          this.moveNodeToThisTreeAndRemoveFromPreviousOne(e, this.tree);\n        } else {\n          this.moveNodeToParentTreeAndRemoveFromPreviousOne(e, this.tree);\n        }\n      })\n    );\n\n    this.subscriptions.push(\n      merge(this.treeService.nodeChecked$, this.treeService.nodeUnchecked$)\n        .pipe(filter((e: NodeCheckedEvent) => this.eventContainsId(e) && this.tree.hasChild(e.node)))\n        .subscribe((e: NodeCheckedEvent) => this.updateCheckboxState())\n    );\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    this.controller = new TreeController(this);\n  }\n\n  public ngOnDestroy(): void {\n    if (get(this.tree, 'node.id', '')) {\n      this.treeService.deleteController(this.tree.node.id);\n    }\n\n    this.subscriptions.forEach(sub => sub && sub.unsubscribe());\n  }\n\n  private swapWithSibling(sibling: Tree, tree: Tree): void {\n    tree.swapWithSibling(sibling);\n    this.treeService.fireNodeMoved(sibling, sibling.parent);\n  }\n\n  private moveNodeToThisTreeAndRemoveFromPreviousOne(e: NodeDraggableEvent, tree: Tree): void {\n    this.treeService.fireNodeRemoved(e.captured.tree);\n    const addedChild = tree.addChild(e.captured.tree);\n    this.treeService.fireNodeMoved(addedChild, e.captured.tree.parent);\n  }\n\n  private moveNodeToParentTreeAndRemoveFromPreviousOne(e: NodeDraggableEvent, tree: Tree): void {\n    this.treeService.fireNodeRemoved(e.captured.tree);\n    const addedSibling = tree.addSibling(e.captured.tree, tree.positionInParent);\n    this.treeService.fireNodeMoved(addedSibling, e.captured.tree.parent);\n  }\n\n  public onNodeSelected(e: { button: number }): void {\n    if (!this.tree.selectionAllowed) {\n      return;\n    }\n\n    if (EventUtils.isLeftButtonClicked(e as MouseEvent)) {\n      this.isSelected = true;\n      this.treeService.fireNodeSelected(this.tree);\n    }\n  }\n\n  public onNodeUnselected(e: { button: number }): void {\n    if (!this.tree.selectionAllowed) {\n      return;\n    }\n\n    if (EventUtils.isLeftButtonClicked(e as MouseEvent)) {\n      this.isSelected = false;\n      this.treeService.fireNodeUnselected(this.tree);\n    }\n  }\n\n  public showRightMenu(e: MouseEvent): void {\n    if (!this.tree.hasRightMenu()) {\n      return;\n    }\n\n    if (EventUtils.isRightButtonClicked(e)) {\n      this.isRightMenuVisible = !this.isRightMenuVisible;\n      this.nodeMenuService.hideMenuForAllNodesExcept(this.nodeElementRef);\n    }\n    e.preventDefault();\n  }\n\n  public showLeftMenu(e: MouseEvent): void {\n    if (!this.tree.hasLeftMenu()) {\n      return;\n    }\n\n    if (EventUtils.isLeftButtonClicked(e)) {\n      this.isLeftMenuVisible = !this.isLeftMenuVisible;\n      this.nodeMenuService.hideMenuForAllNodesExcept(this.nodeElementRef);\n      if (this.isLeftMenuVisible) {\n        e.preventDefault();\n      }\n    }\n  }\n\n  public onMenuItemSelected(e: NodeMenuItemSelectedEvent): void {\n    switch (e.nodeMenuItemAction) {\n      case NodeMenuItemAction.NewTag:\n        this.onNewSelected(e);\n        break;\n      case NodeMenuItemAction.NewFolder:\n        this.onNewSelected(e);\n        break;\n      case NodeMenuItemAction.Rename:\n        this.onRenameSelected();\n        break;\n      case NodeMenuItemAction.Remove:\n        this.onRemoveSelected();\n        break;\n      case NodeMenuItemAction.Custom:\n        this.onCustomSelected();\n        this.treeService.fireMenuItemSelected(this.tree, e.nodeMenuItemSelected);\n        break;\n      default:\n        throw new Error(`Chosen menu item doesn't exist`);\n    }\n  }\n\n  private onNewSelected(e: NodeMenuItemSelectedEvent): void {\n    this.tree.createNode(e.nodeMenuItemAction === NodeMenuItemAction.NewFolder);\n    this.isRightMenuVisible = false;\n    this.isLeftMenuVisible = false;\n  }\n\n  private onRenameSelected(): void {\n    this.tree.markAsBeingRenamed();\n    this.isRightMenuVisible = false;\n    this.isLeftMenuVisible = false;\n  }\n\n  private onRemoveSelected(): void {\n    this.treeService.deleteController(get(this.tree, 'node.id', ''));\n    this.treeService.fireNodeRemoved(this.tree);\n  }\n\n  private onCustomSelected(): void {\n    this.isRightMenuVisible = false;\n    this.isLeftMenuVisible = false;\n  }\n\n  public onSwitchFoldingType(): void {\n    this.tree.switchFoldingType();\n    this.treeService.fireNodeSwitchFoldingType(this.tree);\n  }\n\n  public applyNewValue(e: NodeEditableEvent): void {\n    if ((e.action === NodeEditableEventAction.Cancel || this.tree.isNew()) && Tree.isValueEmpty(e.value)) {\n      return this.treeService.fireNodeRemoved(this.tree);\n    }\n\n    if (this.tree.isNew()) {\n      this.tree.value = e.value;\n      this.treeService.fireNodeCreated(this.tree);\n    }\n\n    if (this.tree.isBeingRenamed()) {\n      const oldValue = this.tree.value;\n      this.tree.value = e.value;\n      this.treeService.fireNodeRenamed(oldValue, this.tree);\n    }\n\n    this.tree.markAsModified();\n  }\n\n  public shouldShowInputForTreeValue(): boolean {\n    return this.tree.isNew() || this.tree.isBeingRenamed();\n  }\n\n  public isRootHidden(): boolean {\n    return this.tree.isRoot() && !this.settings.rootIsVisible;\n  }\n\n  public hasCustomMenu(): boolean {\n    return this.tree.hasCustomMenu();\n  }\n\n  public switchNodeCheckStatus() {\n    if (!this.tree.checked) {\n      this.onNodeChecked();\n    } else {\n      this.onNodeUnchecked();\n    }\n  }\n\n  public onNodeChecked(): void {\n    if (!this.checkboxElementRef) {\n      return;\n    }\n\n    this.checkboxElementRef.nativeElement.indeterminate = false;\n    this.treeService.fireNodeChecked(this.tree);\n    this.executeOnChildController(controller => controller.check());\n    this.tree.checked = true;\n  }\n\n  public onNodeUnchecked(): void {\n    if (!this.checkboxElementRef) {\n      return;\n    }\n\n    this.checkboxElementRef.nativeElement.indeterminate = false;\n    this.treeService.fireNodeUnchecked(this.tree);\n    this.executeOnChildController(controller => controller.uncheck());\n    this.tree.checked = false;\n  }\n\n  private executeOnChildController(executor: (controller: TreeController) => void) {\n    if (this.tree.hasLoadedChildern()) {\n      this.tree.children.forEach((child: Tree) => {\n        const controller = this.treeService.getController(child.id);\n        if (!isNil(controller)) {\n          executor(controller);\n        }\n      });\n    }\n  }\n\n  updateCheckboxState(): void {\n    // Calling setTimeout so the value of isChecked will be updated and after that I'll check the children status.\n    setTimeout(() => {\n      const checkedChildrenAmount = this.tree.checkedChildrenAmount();\n      if (checkedChildrenAmount === 0) {\n        this.checkboxElementRef.nativeElement.indeterminate = false;\n        this.tree.checked = false;\n        this.treeService.fireNodeUnchecked(this.tree);\n      } else if (checkedChildrenAmount === this.tree.loadedChildrenAmount()) {\n        this.checkboxElementRef.nativeElement.indeterminate = false;\n        this.tree.checked = true;\n        this.treeService.fireNodeChecked(this.tree);\n      } else {\n        this.tree.checked = false;\n        this.checkboxElementRef.nativeElement.indeterminate = true;\n        this.treeService.fireNodeIndetermined(this.tree);\n      }\n    });\n  }\n\n  private eventContainsId(event: NodeEvent): boolean {\n    if (!event.node.id) {\n      console.warn(\n        '\"Node with checkbox\" feature requires a unique id assigned to every node, please consider to add it.'\n      );\n      return false;\n    }\n    return true;\n  }\n}\n"]}