30 changed files with 1228 additions and 43 deletions
@ -0,0 +1,53 @@
|
||||
# Angular tree diagram |
||||
|
||||
### About |
||||
This is Angular 2+ Hierarchical UI module. |
||||
|
||||
### Preview |
||||
<img src="http://i.imgur.com/CfQXRGm.png" width="500"> |
||||
|
||||
### Demo |
||||
On [gh-pages](https://artbelikov.github.io/angular2-tree-diagram/) |
||||
|
||||
### Features |
||||
- Drag and drop |
||||
- Zoom and pan |
||||
- Configurable node width/height |
||||
- Add/remove nodes |
||||
- TreeComponent-like UI |
||||
- Pure CSS relation lines |
||||
- No dependencies |
||||
|
||||
### Installation |
||||
``` |
||||
npm i angular2-tree-diagram |
||||
``` |
||||
|
||||
### Usage |
||||
- Import module in your project |
||||
- Use tree-diagram directive |
||||
- Pass array of nodes and config |
||||
- See example.json for more details |
||||
|
||||
### Example |
||||
``` |
||||
<tree-diagram [data]="data"></tree-diagram> |
||||
... |
||||
data = { |
||||
json: [ |
||||
{ |
||||
"guid": "bc4c7a02-5379-4046-92be-12c67af4295a", |
||||
"displayName": "Elentrix", |
||||
"children": [ |
||||
"85d412c2-ebc1-4d56-96c9-7da433ac9bb2", |
||||
"28aac445-83b1-464d-9695-a4157dab6eac" |
||||
] |
||||
}, |
||||
... |
||||
], |
||||
config: { |
||||
nodeWidth: 200, |
||||
nodeHeight: 100 |
||||
} |
||||
} |
||||
``` |
@ -0,0 +1,32 @@
|
||||
// Karma configuration file, see link for more information
|
||||
// https://karma-runner.github.io/1.0/config/configuration-file.html
|
||||
|
||||
module.exports = function (config) { |
||||
config.set({ |
||||
basePath: '', |
||||
frameworks: ['jasmine', '@angular-devkit/build-angular'], |
||||
plugins: [ |
||||
require('karma-jasmine'), |
||||
require('karma-chrome-launcher'), |
||||
require('karma-jasmine-html-reporter'), |
||||
require('karma-coverage-istanbul-reporter'), |
||||
require('@angular-devkit/build-angular/plugins/karma') |
||||
], |
||||
client: { |
||||
clearContext: false // leave Jasmine Spec Runner output visible in browser
|
||||
}, |
||||
coverageIstanbulReporter: { |
||||
dir: require('path').join(__dirname, '../../coverage/ng-tree-diagram'), |
||||
reports: ['html', 'lcovonly', 'text-summary'], |
||||
fixWebpackSourcePaths: true |
||||
}, |
||||
reporters: ['progress', 'kjhtml'], |
||||
port: 9876, |
||||
colors: true, |
||||
logLevel: config.LOG_INFO, |
||||
autoWatch: true, |
||||
browsers: ['Chrome'], |
||||
singleRun: false, |
||||
restartOnFileChange: true |
||||
}); |
||||
}; |
@ -0,0 +1,7 @@
|
||||
{ |
||||
"$schema": "../../node_modules/ng-packagr/ng-package.schema.json", |
||||
"dest": "../../dist/ng-tree-diagram", |
||||
"lib": { |
||||
"entryFile": "src/ng-tree-diagram.ts" |
||||
} |
||||
} |
@ -0,0 +1,15 @@
|
||||
{ |
||||
"name": "angular2-tree-diagram", |
||||
"version": "1.2.0", |
||||
"author": "Artyom Belikov", |
||||
"peerDependencies": { |
||||
"@angular/common": "^9.0.1", |
||||
"@angular/core": "^9.0.1", |
||||
"tslib": "^1.10.0" |
||||
}, |
||||
"license": "MIT", |
||||
"bugs": { |
||||
"url": "https://github.com/artbelikov/angular2-tree-diagram/issues" |
||||
}, |
||||
"homepage": "https://github.com/artbelikov/angular2-tree-diagram#readme" |
||||
} |
@ -0,0 +1,3 @@
|
||||
export * from './node.class'; |
||||
export * from './node-maker.class'; |
||||
export * from './nodes-list.class'; |
@ -0,0 +1,50 @@
|
||||
import { TreeDiagramNode } from './node.class'; |
||||
|
||||
export class TreeDiagramNodeMaker extends TreeDiagramNode { |
||||
private isMakerState = true; |
||||
|
||||
public get isMaker() { |
||||
return this.isMakerState; |
||||
} |
||||
|
||||
public drop(event) { |
||||
event.preventDefault(); |
||||
|
||||
const guid = this.getThisNodeList().draggingNodeGuid; |
||||
|
||||
this.getThisNodeList().rootNode(guid); |
||||
this.displayName = 'New node'; |
||||
|
||||
return false; |
||||
} |
||||
|
||||
public dragenter(event) { |
||||
event.dataTransfer.dropEffect = 'move'; |
||||
|
||||
const guid = this.getThisNodeList().draggingNodeGuid; |
||||
const node = this.getThisNodeList().getNode(guid); |
||||
|
||||
if (node.parentId) { |
||||
this.displayName = 'Root'; |
||||
} |
||||
} |
||||
|
||||
public dragover(event) { |
||||
event.preventDefault(); |
||||
|
||||
const guid = this.getThisNodeList().draggingNodeGuid; |
||||
const node = this.getThisNodeList().getNode(guid); |
||||
|
||||
if (!this.isDragging && node.parentId) { |
||||
this.isDragover = true; |
||||
event.dataTransfer.dropEffect = 'move'; |
||||
} |
||||
|
||||
return false; |
||||
} |
||||
|
||||
public dragleave(event) { |
||||
this.displayName = 'New node'; |
||||
this.isDragover = false; |
||||
} |
||||
} |
@ -0,0 +1,120 @@
|
||||
import { TreeDiagramNodesList } from './nodes-list.class'; |
||||
|
||||
export class TreeDiagramNode { |
||||
public parentId: string | null; |
||||
public guid: string; |
||||
public width: number; |
||||
public height: number; |
||||
public isDragover: boolean; |
||||
public isDragging: boolean; |
||||
public children: Set<string>; |
||||
public displayName: string; |
||||
private toggleState: boolean; |
||||
|
||||
public get isMaker() { |
||||
return false; |
||||
} |
||||
|
||||
constructor( |
||||
props, |
||||
config, |
||||
public getThisNodeList: () => TreeDiagramNodesList |
||||
) { |
||||
if (!props.guid) { |
||||
return; |
||||
} |
||||
|
||||
for (const prop in props) { |
||||
if (props.hasOwnProperty(prop)) { |
||||
this[prop] = props[prop]; |
||||
} |
||||
} |
||||
|
||||
this.toggleState = false; |
||||
|
||||
if (config.nodeWidth) { |
||||
this.width = config.nodeWidth; |
||||
} |
||||
|
||||
if (config.nodeHeight) { |
||||
this.height = config.nodeHeight; |
||||
} |
||||
|
||||
this.children = new Set(props.children as string[]); |
||||
} |
||||
|
||||
public get isExpanded() { |
||||
return this.toggleState; |
||||
} |
||||
|
||||
public destroy() { |
||||
this.getThisNodeList().destroy(this.guid); |
||||
} |
||||
|
||||
public hasChildren() { |
||||
return !!this.children.size; |
||||
} |
||||
|
||||
public toggle(state = !this.toggleState) { |
||||
this.toggleState = state; |
||||
|
||||
if (state) { |
||||
this.getThisNodeList().toggleSiblings(this.guid); |
||||
} |
||||
} |
||||
|
||||
public childrenCount() { |
||||
return this.children.size; |
||||
} |
||||
|
||||
public isRoot() { |
||||
return this.parentId == null; |
||||
} |
||||
|
||||
public dragenter(event) { |
||||
event.dataTransfer.dropEffect = 'move'; |
||||
} |
||||
|
||||
public dragleave(event) { |
||||
this.isDragover = false; |
||||
} |
||||
|
||||
public dragstart(event) { |
||||
event.dataTransfer.effectAllowed = 'move'; |
||||
this.isDragging = true; |
||||
this.toggle(false); |
||||
this.getThisNodeList().draggingNodeGuid = this.guid; |
||||
} |
||||
|
||||
public dragover(event) { |
||||
event.preventDefault(); |
||||
|
||||
if (!this.isDragging) { |
||||
this.isDragover = true; |
||||
} |
||||
|
||||
event.dataTransfer.dropEffect = 'move'; |
||||
return false; |
||||
} |
||||
|
||||
public dragend() { |
||||
this.isDragover = false; |
||||
this.isDragging = false; |
||||
} |
||||
|
||||
public drop(event) { |
||||
event.preventDefault(); |
||||
|
||||
const guid = this.getThisNodeList().draggingNodeGuid; |
||||
|
||||
this.getThisNodeList().transfer(guid, this.guid); |
||||
return false; |
||||
} |
||||
|
||||
public addChild() { |
||||
const newNodeGuid = this.getThisNodeList().newNode(this.guid); |
||||
|
||||
this.children.add(newNodeGuid); |
||||
this.toggle(true); |
||||
} |
||||
} |
@ -0,0 +1,200 @@
|
||||
import { TreeDiagramNode } from './node.class'; |
||||
import { TreeDiagramNodeMaker } from './node-maker.class'; |
||||
|
||||
export class TreeDiagramNodesList { |
||||
public roots: TreeDiagramNode[]; |
||||
public makerGuid: string; |
||||
public draggingNodeGuid; |
||||
private nodesList: Map<string, TreeDiagramNode>; |
||||
private nodeTemplate = { |
||||
displayName: 'New node', |
||||
children: [], |
||||
guid: '', |
||||
parentId: null |
||||
}; |
||||
|
||||
constructor(nodes: any[], private config) { |
||||
this.nodesList = new Map(); |
||||
nodes.forEach(treeNode => { |
||||
this.nodesList.set( |
||||
treeNode.guid, |
||||
new TreeDiagramNode(treeNode, config, this.getThisNodeList.bind(this)) |
||||
); |
||||
}); |
||||
this.makeRoots(); |
||||
this.makerGuid = this.uuidv4(); |
||||
|
||||
const node = { |
||||
guid: this.makerGuid, |
||||
parentId: 'root', |
||||
children: [], |
||||
displayName: 'New node' |
||||
}; |
||||
const maker = new TreeDiagramNodeMaker( |
||||
node, |
||||
this.config, |
||||
this.getThisNodeList.bind(this) |
||||
); |
||||
|
||||
this.nodesList.set(this.makerGuid, maker); |
||||
} |
||||
|
||||
public values() { |
||||
return this.nodesList.values(); |
||||
} |
||||
|
||||
public getNode(guid: string): TreeDiagramNode { |
||||
return this.nodesList.get(guid); |
||||
} |
||||
|
||||
public rootNode(guid: string) { |
||||
const node = this.getNode(guid); |
||||
const maker = this.getNode(this.makerGuid); |
||||
|
||||
node.isDragging = false; |
||||
node.isDragover = false; |
||||
|
||||
if (node.parentId) { |
||||
const parent = this.getNode(node.parentId); |
||||
parent.children.delete(guid); |
||||
} |
||||
|
||||
node.parentId = null; |
||||
|
||||
this.makeRoots(); |
||||
maker.isDragging = false; |
||||
maker.isDragover = false; |
||||
} |
||||
|
||||
public transfer(originId: string, targetId: string) { |
||||
const origin = this.getNode(originId); |
||||
const target = this.getNode(targetId); |
||||
|
||||
origin.isDragover = false; |
||||
origin.isDragging = false; |
||||
target.isDragover = false; |
||||
|
||||
if (origin.parentId === targetId || originId === targetId) { |
||||
return; |
||||
} |
||||
|
||||
const remakeRoots = origin.isRoot(); |
||||
|
||||
if (origin.parentId) { |
||||
const parent = this.getNode(origin.parentId); |
||||
|
||||
parent.children.delete(originId); |
||||
|
||||
if (!parent.hasChildren()) { |
||||
parent.toggle(false); |
||||
} |
||||
} |
||||
|
||||
target.children.add(originId); |
||||
origin.parentId = targetId; |
||||
|
||||
if (remakeRoots) { |
||||
this.makeRoots(); |
||||
} |
||||
|
||||
this.serialize(); |
||||
} |
||||
|
||||
public getThisNodeList() { |
||||
return this; |
||||
} |
||||
|
||||
public toggleSiblings(guid: string) { |
||||
const target = this.getNode(guid); |
||||
|
||||
if (target.parentId) { |
||||
const parent = this.getNode(target.parentId); |
||||
|
||||
parent.children.forEach(nodeGuid => { |
||||
if (nodeGuid === guid) { |
||||
return; |
||||
} |
||||
|
||||
this.getNode(nodeGuid).toggle(false); |
||||
}); |
||||
} else { |
||||
for (const root of this.roots) { |
||||
if (root.guid === guid) { |
||||
continue; |
||||
} |
||||
|
||||
root.toggle(false); |
||||
} |
||||
} |
||||
} |
||||
|
||||
public serialize() { |
||||
const out = []; |
||||
|
||||
this.nodesList.forEach((node: TreeDiagramNode) => { |
||||
const json: any = { |
||||
guid: node.guid, |
||||
displayName: node.displayName, |
||||
parentId: node.parentId, |
||||
children: Array.from(node.children), |
||||
}; |
||||
|
||||
out.push(json); |
||||
}); |
||||
|
||||
return out; |
||||
} |
||||
|
||||
public destroy(guid: string) { |
||||
const target = this.getNode(guid); |
||||
|
||||
if (target.parentId) { |
||||
const parent = this.getNode(target.parentId); |
||||
parent.children.delete(guid); |
||||
} |
||||
|
||||
if (target.hasChildren()) { |
||||
target.children.forEach((child: string) => { |
||||
this.nodesList.delete(child); |
||||
}); |
||||
} |
||||
|
||||
this.nodesList.delete(guid); |
||||
} |
||||
|
||||
public newNode(parentId = null) { |
||||
const nodeTemplate = Object.assign({}, this.nodeTemplate); |
||||
|
||||
nodeTemplate.guid = this.uuidv4(); |
||||
nodeTemplate.parentId = parentId; |
||||
this.nodesList.set( |
||||
nodeTemplate.guid, |
||||
new TreeDiagramNode( |
||||
nodeTemplate, |
||||
this.config, |
||||
this.getThisNodeList.bind(this) |
||||
) |
||||
); |
||||
this.makeRoots(); |
||||
|
||||
return nodeTemplate.guid; |
||||
} |
||||
|
||||
private uuidv4() { |
||||
// tslint:disable-next-line:only-arrow-functions
|
||||
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) { |
||||
// tslint:disable-next-line:one-variable-per-declaration no-bitwise
|
||||
const r = (Math.random() * 16) | 0, |
||||
// tslint:disable-next-line:triple-equals no-bitwise
|
||||
v = c == 'x' ? r : (r & 0x3) | 0x8; |
||||
|
||||
return v.toString(16); |
||||
}); |
||||
} |
||||
|
||||
private makeRoots() { |
||||
this.roots = Array.from(this.values()).filter((node: TreeDiagramNode) => |
||||
node.isRoot() |
||||
); |
||||
} |
||||
} |
@ -0,0 +1 @@
|
||||
export { TreeDiagramModule } from './tree.module'; |
@ -0,0 +1 @@
|
||||
export { NodeComponent } from './node.component'; |
@ -0,0 +1,65 @@
|
||||
<div class="line-to"></div> |
||||
<div |
||||
id="tree-node-{{ node.guid }}" |
||||
[ngClass]="{ |
||||
'tree-element-container': !0, |
||||
'tree-element-has-children': node.hasChildren() |
||||
}" |
||||
[style.width.px]="node.width" |
||||
[style.height.px]="node.height" |
||||
> |
||||
<div |
||||
[ngClass]="{ |
||||
'tree-element-main': !0, |
||||
dragover: node.isDragover, |
||||
expanded: node.isExpanded, |
||||
dragging: node.isDragging, |
||||
'tree-new-node': node.isMaker |
||||
}" |
||||
[attr.draggable]="node.isMaker ? null : 'true'" |
||||
[style.width.px]="node.width" |
||||
[style.height.px]="node.height" |
||||
(drop)="node.drop($event)" |
||||
(dragenter)="node.dragenter($event)" |
||||
(dragstart)="node.dragstart($event)" |
||||
(dragover)="node.dragover($event)" |
||||
(dragend)="node.dragend()" |
||||
(dragleave)="node.dragleave($event)" |
||||
> |
||||
<div |
||||
class="rect" |
||||
[style.width.px]="node.width" |
||||
[style.height.px]="node.height" |
||||
> |
||||
<div class="buttons"> |
||||
<div class="delete" (click)="node.destroy()"></div> |
||||
<div class="toggler" (click)="node.toggle()"></div> |
||||
<div class="add" (click)="node.addChild()"></div> |
||||
</div> |
||||
<div class="tree-text tree-text-non-editable"> |
||||
<span>{{ node.displayName }}</span> |
||||
</div> |
||||
<div class="tree-text tree-text-editable"> |
||||
<span |
||||
contenteditable |
||||
[innerHtml]="node.displayName" |
||||
(blur)="onNodeBlur($event, node.guid)" |
||||
></span> |
||||
<span class="children-count"> ({{ node.childrenCount() }})</span> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
<div |
||||
class="tree-children" |
||||
[style.transform]="childrenTransform" |
||||
*ngIf="node.isExpanded" |
||||
> |
||||
<div class="tree-elements-group"> |
||||
<tree-diagram-node |
||||
*ngFor="let child of node.children" |
||||
[nodeId]="child" |
||||
class="tree-node tree-child" |
||||
></tree-diagram-node> |
||||
</div> |
||||
</div> |
||||
</div> |
@ -0,0 +1,253 @@
|
||||
@mixin tree-button { |
||||
width: 20px; |
||||
height: 20px; |
||||
cursor: pointer; |
||||
border-radius: 2px; |
||||
} |
||||
|
||||
.toggler { |
||||
position: absolute; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: -10px; |
||||
background: #2c4c63; |
||||
margin: 0 auto; |
||||
display: none; |
||||
z-index: 10; |
||||
@include tree-button; |
||||
} |
||||
|
||||
.children-count { |
||||
display: none; |
||||
} |
||||
|
||||
.tree-element-has-children { |
||||
> .tree-element-main { |
||||
.toggler { |
||||
display: block; |
||||
} |
||||
.rect { |
||||
} |
||||
.children-count { |
||||
display: inline; |
||||
} |
||||
} |
||||
> .tree-children { |
||||
display: inline-block; |
||||
} |
||||
} |
||||
|
||||
.rect { |
||||
position: relative; |
||||
background-color: #fafafa !important; |
||||
border: 1px solid #dadada; |
||||
box-sizing: border-box; |
||||
-webkit-print-color-adjust: exact; |
||||
cursor: default !important; |
||||
display: flex; |
||||
justify-content: center; |
||||
align-items: center; |
||||
font-size: 15px; |
||||
border-radius: 2px; |
||||
} |
||||
|
||||
.tree-element-main { |
||||
text-align: center; |
||||
margin: 0 auto; |
||||
&:hover { |
||||
.buttons { |
||||
display: block; |
||||
} |
||||
} |
||||
&.expanded { |
||||
.rect { |
||||
background-color: #bce5ff !important; |
||||
} |
||||
.toggler { |
||||
transform: rotateZ(-45deg); |
||||
background: #427396; |
||||
} |
||||
} |
||||
&.dragover { |
||||
.rect { |
||||
box-shadow: 0 0 5px #427396; |
||||
} |
||||
} |
||||
&.dragging { |
||||
.buttons { |
||||
display: none !important; |
||||
} |
||||
.tree-node:before { |
||||
display: none !important; |
||||
} |
||||
} |
||||
} |
||||
|
||||
.tree-element-container { |
||||
z-index: 100; |
||||
} |
||||
|
||||
.tree-children { |
||||
text-align: center; |
||||
display: inline-block; |
||||
position: relative; |
||||
white-space: nowrap; |
||||
perspective: 3000px; |
||||
perspective-origin: center bottom; |
||||
&:before { |
||||
content: ""; |
||||
width: calc(50% - 1px); |
||||
position: absolute; |
||||
height: 30px; |
||||
left: 0; |
||||
top: -45px; |
||||
border-right: 1px solid #dadada; |
||||
max-width: 100%; |
||||
max-height: 100%; |
||||
} |
||||
} |
||||
|
||||
.line-to { |
||||
position: absolute; |
||||
top: -30px; |
||||
border-top: 1px solid #dadada; |
||||
width: calc(100% + 30px); |
||||
display: none; |
||||
} |
||||
|
||||
.tree-node { |
||||
position: relative; |
||||
display: inline-block; |
||||
margin: 15px; |
||||
vertical-align: top; |
||||
&:before { |
||||
content: ""; |
||||
width: calc(50% - 1px); |
||||
position: absolute; |
||||
height: 30px; |
||||
left: 0; |
||||
top: -30px; |
||||
border-right: 1px solid #dadada; |
||||
} |
||||
&:only-of-type { |
||||
> .line-to { |
||||
display: none !important; |
||||
} |
||||
} |
||||
} |
||||
|
||||
.buttons { |
||||
position: absolute; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
display: none; |
||||
.delete { |
||||
@include tree-button; |
||||
background-color: #a34851; |
||||
position: absolute; |
||||
right: -10px; |
||||
top: -10px; |
||||
} |
||||
.add { |
||||
@include tree-button; |
||||
background-color: #256947; |
||||
position: absolute; |
||||
right: -10px; |
||||
bottom: -10px; |
||||
&:before { |
||||
content: ""; |
||||
position: absolute; |
||||
height: 12px; |
||||
width: 4px; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
margin: auto; |
||||
background-color: #2ba423; |
||||
} |
||||
&:after { |
||||
content: ""; |
||||
position: absolute; |
||||
width: 12px; |
||||
height: 4px; |
||||
top: 0; |
||||
left: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
margin: auto; |
||||
background-color: #2ba423; |
||||
} |
||||
} |
||||
} |
||||
|
||||
.tree-text { |
||||
z-index: 10; |
||||
white-space: pre-line; |
||||
span { |
||||
cursor: pointer; |
||||
} |
||||
} |
||||
|
||||
.tree-elements-group { |
||||
position: relative; |
||||
|
||||
& > .tree-node.tree-child { |
||||
& > .line-to { |
||||
left: 0; |
||||
display: block; |
||||
} |
||||
&:first-of-type { |
||||
& > .line-to { |
||||
right: -30px; |
||||
width: calc(50% + 30px); |
||||
display: block; |
||||
left: auto; |
||||
} |
||||
} |
||||
&:last-of-type { |
||||
& > .line-to { |
||||
left: 0; |
||||
right: auto; |
||||
width: 50%; |
||||
display: block; |
||||
} |
||||
} |
||||
> .tree-child:last-child { |
||||
margin-right: 0; |
||||
} |
||||
> .tree-child:first-child { |
||||
margin-left: 0; |
||||
} |
||||
} |
||||
} |
||||
|
||||
.tree-text-non-editable { |
||||
display: none; |
||||
} |
||||
|
||||
.tree-new-node { |
||||
.rect { |
||||
opacity: 0.5; |
||||
border: 1px dashed #dadada; |
||||
cursor: pointer !important; |
||||
} |
||||
&:hover, |
||||
&.dragover { |
||||
.rect { |
||||
opacity: 1; |
||||
} |
||||
} |
||||
.tree-children, |
||||
.buttons { |
||||
display: none !important; |
||||
} |
||||
.tree-text-non-editable { |
||||
display: block; |
||||
} |
||||
.tree-text-editable { |
||||
display: none; |
||||
} |
||||
} |
@ -0,0 +1,49 @@
|
||||
import { Component, Input } from '@angular/core'; |
||||
import { DomSanitizer } from '@angular/platform-browser'; |
||||
|
||||
import { NodesListService } from '../services/nodes-list.service'; |
||||
import { TreeDiagramNode } from '../classes/node.class'; |
||||
import { TreeDiagramNodeMaker } from '../classes/node-maker.class'; |
||||
|
||||
@Component({ |
||||
// tslint:disable-next-line:component-selector
|
||||
selector: 'tree-diagram-node', |
||||
styleUrls: ['./node.component.scss'], |
||||
templateUrl: './node.component.html' |
||||
}) |
||||
export class NodeComponent { |
||||
public node: TreeDiagramNode | TreeDiagramNodeMaker; |
||||
public childrenTransform; |
||||
private readonly isRtl: boolean; |
||||
|
||||
constructor( |
||||
private nodesSrv: NodesListService, |
||||
private sanitizer: DomSanitizer |
||||
) { |
||||
this.isRtl = document.getElementsByTagName('html')[0].getAttribute('dir') === 'rtl'; |
||||
} |
||||
|
||||
@Input() set nodeId(guid) { |
||||
this.node = this.nodesSrv.getNode(guid); |
||||
|
||||
let calculation = `translate(calc(-50% + ${Math.round( |
||||
this.node.width / 2 |
||||
)}px), 45px)`;
|
||||
|
||||
if (this.isRtl) { |
||||
calculation = `translate(calc(50% - ${Math.round( |
||||
this.node.width / 2 |
||||
)}px), 45px)`;
|
||||
} |
||||
|
||||
this.childrenTransform = this.sanitizer.bypassSecurityTrustStyle( |
||||
calculation |
||||
); |
||||
} |
||||
|
||||
onNodeBlur(event, nodeId) { |
||||
const node = this.nodesSrv.getNode(nodeId); |
||||
|
||||
node.displayName = event.target.innerText; |
||||
} |
||||
} |
@ -0,0 +1,13 @@
|
||||
html[dir=rtl] .tree-elements-group > div > .line-to { |
||||
right: 0!important; |
||||
} |
||||
|
||||
html[dir=rtl] .tree-elements-group > div:first-of-type > .line-to { |
||||
left: -30px!important; |
||||
right: auto!important; |
||||
} |
||||
|
||||
html[dir=rtl] .tree-elements-group > div:last-of-type > .line-to { |
||||
right: 0!important; |
||||
left: auto; |
||||
} |
@ -0,0 +1,29 @@
|
||||
import { Injectable } from '@angular/core'; |
||||
import { TreeDiagramNodesList } from '../classes'; |
||||
|
||||
@Injectable() |
||||
export class NodesListService { |
||||
private nodesList: TreeDiagramNodesList; |
||||
|
||||
public loadNodes(nodes: any[], config) { |
||||
this.nodesList = new TreeDiagramNodesList(nodes, config); |
||||
|
||||
return this.nodesList; |
||||
} |
||||
|
||||
public getNode(guid) { |
||||
return this.nodesList.getNode(guid); |
||||
} |
||||
|
||||
public newNode() { |
||||
this.nodesList.newNode(); |
||||
} |
||||
|
||||
public makerNode() { |
||||
return this.nodesList.makerGuid; |
||||
} |
||||
|
||||
public toJsonString() { |
||||
return JSON.stringify(this.nodesList.serialize()); |
||||
} |
||||
} |
@ -0,0 +1,27 @@
|
||||
<div |
||||
class="tree-pane" |
||||
(mousedown)="onmousedown()" |
||||
(mouseup)="onmouseup()" |
||||
(mousemove)="onmousemove($event)" |
||||
(mousewheel)="onmousewheel($event)" |
||||
> |
||||
<div class="tree-paning-container" [style.transform]="paneTransform"> |
||||
<div *ngIf="nodes" class="tree-roots-elements"> |
||||
|
||||
<tree-diagram-node |
||||
[nodeId]="node.guid" |
||||
class="tree-root tree-node" |
||||
(mousedown)="preventMouse($event)" |
||||
*ngFor="let node of nodes.roots" |
||||
></tree-diagram-node> |
||||
|
||||
<tree-diagram-node |
||||
[nodeId]="nodeMaker" |
||||
(click)="newNode()" |
||||
(mousedown)="preventMouse($event)" |
||||
class="tree-root tree-new-node tree-node" |
||||
></tree-diagram-node> |
||||
|
||||
</div> |
||||
</div> |
||||
</div> |
@ -0,0 +1,41 @@
|
||||
:host { |
||||
-webkit-print-color-adjust: exact; |
||||
position: relative; |
||||
display: block; |
||||
-webkit-touch-callout: none; |
||||
user-select: none; |
||||
overflow: hidden; |
||||
height: 100vh; |
||||
text-align: center; |
||||
} |
||||
|
||||
.tree-roots-elements { |
||||
position: relative; |
||||
text-align: center; |
||||
display: inline-block; |
||||
white-space: nowrap; |
||||
cursor: default !important; |
||||
font-size: 0; |
||||
transform-origin: center; |
||||
} |
||||
|
||||
.tree-node { |
||||
position: relative; |
||||
display: inline-block; |
||||
margin: 15px; |
||||
vertical-align: top; |
||||
&:only-of-type { |
||||
> .line-to { |
||||
display: none; |
||||
} |
||||
} |
||||
} |
||||
|
||||
.tree-pane, |
||||
.tree-paning-container { |
||||
position: absolute; |
||||
left: 0; |
||||
top: 0; |
||||
right: 0; |
||||
bottom: 0; |
||||
} |
@ -0,0 +1,95 @@
|
||||
import { Component, Input } from '@angular/core'; |
||||
import { DomSanitizer } from '@angular/platform-browser'; |
||||
|
||||
import { NodesListService } from './services/nodes-list.service'; |
||||
|
||||
@Component({ |
||||
// tslint:disable-next-line:component-selector
|
||||
selector: 'tree-diagram', |
||||
styleUrls: ['./tree.component.scss'], |
||||
templateUrl: './tree.component.html' |
||||
}) |
||||
export class TreeComponent { |
||||
public nodes; |
||||
private config = { |
||||
nodeWidth: 200, |
||||
nodeHeight: 100 |
||||
}; |
||||
private paneDragging = false; |
||||
private paneTransformState; |
||||
private zoom = 1; |
||||
private paneX = 0; |
||||
private paneY = 0; |
||||
|
||||
public get paneTransform() { |
||||
return this.paneTransformState; |
||||
} |
||||
|
||||
public set paneTransform(value) { |
||||
this.paneTransformState = value; |
||||
} |
||||
|
||||
constructor( |
||||
private nodesSrv: NodesListService, |
||||
private sanitizer: DomSanitizer |
||||
) {} |
||||
|
||||
@Input() set data(data: { config: any; json: any[] }) { |
||||
if (!data || !Array.isArray(data.json)) { |
||||
return; |
||||
} |
||||
|
||||
if (typeof data.config === 'object') { |
||||
this.config = Object.assign(this.config, data.config); |
||||
} |
||||
|
||||
this.nodes = this.nodesSrv.loadNodes(data.json, this.config); |
||||
} |
||||
|
||||
public get nodeMaker() { |
||||
return this.nodesSrv.makerNode(); |
||||
} |
||||
|
||||
public newNode() { |
||||
this.nodesSrv.newNode(); |
||||
} |
||||
|
||||
public onmousedown() { |
||||
this.paneDragging = true; |
||||
} |
||||
|
||||
public onmousemove(event) { |
||||
if (this.paneDragging) { |
||||
const { movementX, movementY } = event; |
||||
|
||||
this.paneX += movementX; |
||||
this.paneY += movementY; |
||||
this.makeTransform(); |
||||
} |
||||
} |
||||
|
||||
public onmouseup() { |
||||
this.paneDragging = false; |
||||
} |
||||
|
||||
public makeTransform() { |
||||
this.paneTransform = this.sanitizer.bypassSecurityTrustStyle( |
||||
`translate(${this.paneX}px, ${this.paneY}px) scale(${this.zoom})` |
||||
); |
||||
} |
||||
|
||||
public preventMouse(event) { |
||||
event.stopPropagation(); |
||||
} |
||||
|
||||
public onmousewheel(event) { |
||||
let delta; |
||||
|
||||
event.preventDefault(); |
||||
delta = event.detail || event.wheelDelta; |
||||
this.zoom += delta / 1000 / 2; |
||||
this.zoom = Math.min(Math.max(this.zoom, 0.2), 3); |
||||
|
||||
this.makeTransform(); |
||||
} |
||||
} |
@ -0,0 +1,26 @@
|
||||
import { CommonModule } from '@angular/common'; |
||||
import { NgModule } from '@angular/core'; |
||||
|
||||
import { TreeComponent } from './tree.component'; |
||||
import { NodeComponent } from './node'; |
||||
import { NodesListService } from './services/nodes-list.service'; |
||||
|
||||
@NgModule({ |
||||
declarations: [ |
||||
TreeComponent, |
||||
NodeComponent |
||||
], |
||||
imports: [ |
||||
CommonModule |
||||
], |
||||
exports: [ |
||||
TreeComponent, |
||||
NodeComponent |
||||
], |
||||
providers: [ |
||||
NodesListService |
||||
] |
||||
}) |
||||
export class TreeDiagramModule { |
||||
|
||||
} |
@ -0,0 +1,8 @@
|
||||
/* |
||||
* Public API Surface of ng-tree-diagram |
||||
*/ |
||||
|
||||
export * from './lib/tree.module'; |
||||
export * from './lib/services/nodes-list.service'; |
||||
export * from './lib/node'; |
||||
export * from './lib/classes'; |
@ -0,0 +1,26 @@
|
||||
// This file is required by karma.conf.js and loads recursively all the .spec and framework files
|
||||
|
||||
import 'zone.js/dist/zone'; |
||||
import 'zone.js/dist/zone-testing'; |
||||
import { getTestBed } from '@angular/core/testing'; |
||||
import { |
||||
BrowserDynamicTestingModule, |
||||
platformBrowserDynamicTesting |
||||
} from '@angular/platform-browser-dynamic/testing'; |
||||
|
||||
declare const require: { |
||||
context(path: string, deep?: boolean, filter?: RegExp): { |
||||
keys(): string[]; |
||||
<T>(id: string): T; |
||||
}; |
||||
}; |
||||
|
||||
// First, initialize the Angular testing environment.
|
||||
getTestBed().initTestEnvironment( |
||||
BrowserDynamicTestingModule, |
||||
platformBrowserDynamicTesting() |
||||
); |
||||
// Then we find all the tests.
|
||||
const context = require.context('./', true, /\.spec\.ts$/); |
||||
// And load the modules.
|
||||
context.keys().map(context); |
@ -0,0 +1,23 @@
|
||||
{ |
||||
"extends": "../../tsconfig.json", |
||||
"compilerOptions": { |
||||
"outDir": "../../out-tsc/lib", |
||||
"target": "es2015", |
||||
"declaration": true, |
||||
"inlineSources": true, |
||||
"types": [], |
||||
"lib": [ |
||||
"dom", |
||||
"es2018" |
||||
] |
||||
}, |
||||
"angularCompilerOptions": { |
||||
"skipTemplateCodegen": true, |
||||
"strictMetadataEmit": true, |
||||
"enableResourceInlining": true |
||||
}, |
||||
"exclude": [ |
||||
"src/test.ts", |
||||
"**/*.spec.ts" |
||||
] |
||||
} |
@ -0,0 +1,6 @@
|
||||
{ |
||||
"extends": "./tsconfig.lib.json", |
||||
"angularCompilerOptions": { |
||||
"enableIvy": false |
||||
} |
||||
} |
@ -0,0 +1,17 @@
|
||||
{ |
||||
"extends": "../../tsconfig.json", |
||||
"compilerOptions": { |
||||
"outDir": "../../out-tsc/spec", |
||||
"types": [ |
||||
"jasmine", |
||||
"node" |
||||
] |
||||
}, |
||||
"files": [ |
||||
"src/test.ts" |
||||
], |
||||
"include": [ |
||||
"**/*.spec.ts", |
||||
"**/*.d.ts" |
||||
] |
||||
} |
@ -0,0 +1,17 @@
|
||||
{ |
||||
"extends": "../../tslint.json", |
||||
"rules": { |
||||
"directive-selector": [ |
||||
true, |
||||
"attribute", |
||||
"lib", |
||||
"camelCase" |
||||
], |
||||
"component-selector": [ |
||||
true, |
||||
"element", |
||||
"lib", |
||||
"kebab-case" |
||||
] |
||||
} |
||||
} |
Loading…
Reference in new issue