angular-tree-component
Version:
A simple yet powerful tree component for Angular2
120 lines (119 loc) • 15.5 kB
JavaScript
import { Component, ContentChild, EventEmitter, HostListener, Input, Output, TemplateRef, ViewChild } from '@angular/core';
import { TreeModel } from '../models/tree.model';
import { TreeDraggedElement } from '../models/tree-dragged-element.model';
import { TreeOptions } from '../models/tree-options.model';
import { TreeViewportComponent } from './tree-viewport.component';
import includes from 'lodash/includes';
import pick from 'lodash/pick';
var TreeComponent = /** @class */ (function () {
function TreeComponent(treeModel, treeDraggedElement) {
var _this = this;
this.treeModel = treeModel;
this.treeDraggedElement = treeDraggedElement;
treeModel.eventNames.forEach(function (name) { return _this[name] = new EventEmitter(); });
treeModel.subscribeToState(function (state) { return _this.stateChange.emit(state); });
}
Object.defineProperty(TreeComponent.prototype, "nodes", {
// Will be handled in ngOnChanges
set: function (nodes) {
},
enumerable: true,
configurable: true
});
;
Object.defineProperty(TreeComponent.prototype, "options", {
set: function (options) {
},
enumerable: true,
configurable: true
});
;
Object.defineProperty(TreeComponent.prototype, "focused", {
set: function (value) {
this.treeModel.setFocus(value);
},
enumerable: true,
configurable: true
});
Object.defineProperty(TreeComponent.prototype, "state", {
set: function (state) {
this.treeModel.setState(state);
},
enumerable: true,
configurable: true
});
TreeComponent.prototype.onKeydown = function ($event) {
if (!this.treeModel.isFocused)
return;
if (includes(['input', 'textarea'], document.activeElement.tagName.toLowerCase()))
return;
var focusedNode = this.treeModel.getFocusedNode();
this.treeModel.performKeyAction(focusedNode, $event);
};
TreeComponent.prototype.onMousedown = function ($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);
}
};
TreeComponent.prototype.ngOnChanges = function (changes) {
if (changes.options || changes.nodes) {
this.treeModel.setData({
options: changes.options && changes.options.currentValue,
nodes: changes.nodes && changes.nodes.currentValue,
events: pick(this, this.treeModel.eventNames)
});
}
};
TreeComponent.prototype.sizeChanged = function () {
this.viewportComponent.setViewport();
};
TreeComponent.decorators = [
{ type: Component, args: [{
selector: 'Tree, tree-root',
providers: [TreeModel],
styles: [],
template: "\n <tree-viewport #viewport>\n <div\n class=\"angular-tree-component\"\n [class.node-dragging]=\"treeDraggedElement.isDragging()\"\n [class.angular-tree-component-rtl]=\"treeModel.options.rtl\">\n <tree-node-collection\n *ngIf=\"treeModel.roots\"\n [nodes]=\"treeModel.roots\"\n [treeModel]=\"treeModel\"\n [templates]=\"{\n loadingTemplate: loadingTemplate,\n treeNodeTemplate: treeNodeTemplate,\n treeNodeWrapperTemplate: treeNodeWrapperTemplate,\n treeNodeFullTemplate: treeNodeFullTemplate\n }\">\n </tree-node-collection>\n <tree-node-drop-slot\n class=\"empty-tree-drop-slot\"\n *ngIf=\"treeModel.isEmptyTree()\"\n [dropIndex]=\"0\"\n [node]=\"treeModel.virtualRoot\">\n </tree-node-drop-slot>\n </div>\n </tree-viewport>\n "
},] },
];
/** @nocollapse */
TreeComponent.ctorParameters = function () { return [
{ type: TreeModel },
{ type: TreeDraggedElement }
]; };
TreeComponent.propDecorators = {
loadingTemplate: [{ type: ContentChild, args: ['loadingTemplate',] }],
treeNodeTemplate: [{ type: ContentChild, args: ['treeNodeTemplate',] }],
treeNodeWrapperTemplate: [{ type: ContentChild, args: ['treeNodeWrapperTemplate',] }],
treeNodeFullTemplate: [{ type: ContentChild, args: ['treeNodeFullTemplate',] }],
viewportComponent: [{ type: ViewChild, args: ['viewport',] }],
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'],] }]
};
return TreeComponent;
}());
export { TreeComponent };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree.component.js","sourceRoot":"","sources":["../../lib/components/tree.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, ContentChild, EventEmitter, HostListener, Input, OnChanges, Output, TemplateRef, ViewChild } from '@angular/core';\nimport { TreeModel } from '../models/tree.model';\nimport { TreeDraggedElement } from '../models/tree-dragged-element.model';\nimport { TreeOptions } from '../models/tree-options.model';\nimport { TreeViewportComponent } from './tree-viewport.component';\n\nimport includes from 'lodash/includes';\nimport pick from 'lodash/pick';\n\n@Component({\n  selector: 'Tree, tree-root',\n  providers: [TreeModel],\n  styles: [],\n  template: `\n      <tree-viewport #viewport>\n          <div\n                  class=\"angular-tree-component\"\n                  [class.node-dragging]=\"treeDraggedElement.isDragging()\"\n                  [class.angular-tree-component-rtl]=\"treeModel.options.rtl\">\n              <tree-node-collection\n                      *ngIf=\"treeModel.roots\"\n                      [nodes]=\"treeModel.roots\"\n                      [treeModel]=\"treeModel\"\n                      [templates]=\"{\n            loadingTemplate: loadingTemplate,\n            treeNodeTemplate: treeNodeTemplate,\n            treeNodeWrapperTemplate: treeNodeWrapperTemplate,\n            treeNodeFullTemplate: treeNodeFullTemplate\n          }\">\n              </tree-node-collection>\n              <tree-node-drop-slot\n                      class=\"empty-tree-drop-slot\"\n                      *ngIf=\"treeModel.isEmptyTree()\"\n                      [dropIndex]=\"0\"\n                      [node]=\"treeModel.virtualRoot\">\n              </tree-node-drop-slot>\n          </div>\n      </tree-viewport>\n  `\n})\nexport class TreeComponent implements OnChanges {\n  _nodes: any[];\n  _options: TreeOptions;\n\n  @ContentChild('loadingTemplate') loadingTemplate: TemplateRef<any>;\n  @ContentChild('treeNodeTemplate') treeNodeTemplate: TemplateRef<any>;\n  @ContentChild('treeNodeWrapperTemplate') treeNodeWrapperTemplate: TemplateRef<any>;\n  @ContentChild('treeNodeFullTemplate') treeNodeFullTemplate: TemplateRef<any>;\n  @ViewChild('viewport') viewportComponent: TreeViewportComponent;\n\n  // Will be handled in ngOnChanges\n  @Input() set nodes(nodes: any[]) {\n  };\n\n  @Input() set options(options: TreeOptions) {\n  };\n\n  @Input() set focused(value: boolean) {\n    this.treeModel.setFocus(value);\n  }\n\n  @Input() set state(state) {\n    this.treeModel.setState(state);\n  }\n\n  @Output() toggleExpanded;\n  @Output() activate;\n  @Output() deactivate;\n  @Output() nodeActivate;\n  @Output() nodeDeactivate;\n  @Output() select;\n  @Output() deselect;\n  @Output() focus;\n  @Output() blur;\n  @Output() updateData;\n  @Output() initialized;\n  @Output() moveNode;\n  @Output() copyNode;\n  @Output() loadNodeChildren;\n  @Output() changeFilter;\n  @Output() event;\n  @Output() stateChange;\n\n  constructor(\n    public treeModel: TreeModel,\n    public treeDraggedElement: TreeDraggedElement) {\n\n    treeModel.eventNames.forEach((name) => this[name] = new EventEmitter());\n    treeModel.subscribeToState((state) => this.stateChange.emit(state));\n  }\n\n  @HostListener('body: keydown', ['$event'])\n  onKeydown($event) {\n    if (!this.treeModel.isFocused) return;\n    if (includes(['input', 'textarea'],\n      document.activeElement.tagName.toLowerCase())) return;\n\n    const focusedNode = this.treeModel.getFocusedNode();\n\n    this.treeModel.performKeyAction(focusedNode, $event);\n  }\n\n  @HostListener('body: mousedown', ['$event'])\n  onMousedown($event) {\n    function isOutsideClick(startElement: Element, nodeName: string) {\n      return !startElement ? true : startElement.localName === nodeName ? false : isOutsideClick(startElement.parentElement, nodeName);\n    }\n\n    if (isOutsideClick($event.target, 'tree-root')) {\n      this.treeModel.setFocus(false);\n    }\n  }\n\n  ngOnChanges(changes) {\n    if (changes.options || changes.nodes) {\n      this.treeModel.setData({\n        options: changes.options && changes.options.currentValue,\n        nodes: changes.nodes && changes.nodes.currentValue,\n        events: pick(this, this.treeModel.eventNames)\n      });\n    }\n  }\n\n  sizeChanged() {\n    this.viewportComponent.setViewport();\n  }\n}\n"]}