mirror of
https://github.com/richard-loafle/fuse-angular.git
synced 2025-04-19 23:02:33 +00:00
132 lines
3.5 KiB
TypeScript
Executable File
132 lines
3.5 KiB
TypeScript
Executable File
import {NestedTreeControl} from '@angular/cdk/tree';
|
|
import {Component, Injectable} from '@angular/core';
|
|
import {MatTreeNestedDataSource} from '@angular/material/tree';
|
|
import {BehaviorSubject, of as observableOf} from 'rxjs';
|
|
|
|
/**
|
|
* Json node data with nested structure. Each node has a filename and a value or a list of children
|
|
*/
|
|
export class FileNode {
|
|
children: FileNode[];
|
|
filename: string;
|
|
type: any;
|
|
}
|
|
|
|
/**
|
|
* The Json tree data in string. The data could be parsed into Json object
|
|
*/
|
|
const TREE_DATA = `
|
|
{
|
|
"Documents": {
|
|
"angular": {
|
|
"src": {
|
|
"core": "ts",
|
|
"compiler": "ts"
|
|
}
|
|
},
|
|
"material2": {
|
|
"src": {
|
|
"button": "ts",
|
|
"checkbox": "ts",
|
|
"input": "ts"
|
|
}
|
|
}
|
|
},
|
|
"Downloads": {
|
|
"Tutorial": "html",
|
|
"November": "pdf",
|
|
"October": "pdf"
|
|
},
|
|
"Pictures": {
|
|
"Sun": "png",
|
|
"Woods": "jpg",
|
|
"Photo Booth Library": {
|
|
"Contents": "dir",
|
|
"Pictures": "dir"
|
|
}
|
|
},
|
|
"Applications": {
|
|
"Chrome": "app",
|
|
"Calendar": "app",
|
|
"Webstorm": "app"
|
|
}
|
|
}`;
|
|
|
|
/**
|
|
* File database, it can build a tree structured Json object from string.
|
|
* Each node in Json object represents a file or a directory. For a file, it has filename and type.
|
|
* For a directory, it has filename and children (a list of files or directories).
|
|
* The input will be a json object string, and the output is a list of `FileNode` with nested
|
|
* structure.
|
|
*/
|
|
@Injectable()
|
|
export class FileDatabase {
|
|
dataChange: BehaviorSubject<FileNode[]> = new BehaviorSubject<FileNode[]>([]);
|
|
|
|
get data(): FileNode[] { return this.dataChange.value; }
|
|
|
|
constructor() {
|
|
this.initialize();
|
|
}
|
|
|
|
initialize() {
|
|
// Parse the string to json object.
|
|
const dataObject = JSON.parse(TREE_DATA);
|
|
|
|
// Build the tree nodes from Json object. The result is a list of `FileNode` with nested
|
|
// file node as children.
|
|
const data = this.buildFileTree(dataObject, 0);
|
|
|
|
// Notify the change.
|
|
this.dataChange.next(data);
|
|
}
|
|
|
|
/**
|
|
* Build the file structure tree. The `value` is the Json object, or a sub-tree of a Json object.
|
|
* The return value is the list of `FileNode`.
|
|
*/
|
|
buildFileTree(value: any, level: number): FileNode[] {
|
|
let data: any[] = [];
|
|
for (let k in value) {
|
|
let v = value[k];
|
|
let node = new FileNode();
|
|
node.filename = `${k}`;
|
|
if (v === null || v === undefined) {
|
|
// no action
|
|
} else if (typeof v === 'object') {
|
|
node.children = this.buildFileTree(v, level + 1);
|
|
} else {
|
|
node.type = v;
|
|
}
|
|
data.push(node);
|
|
}
|
|
return data;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @title Tree with nested nodes
|
|
*/
|
|
@Component({
|
|
selector: 'cdk-tree-nested-example',
|
|
templateUrl: 'cdk-tree-nested-example.html',
|
|
styleUrls: ['cdk-tree-nested-example.css'],
|
|
providers: [FileDatabase]
|
|
})
|
|
export class CdkTreeNestedExample {
|
|
nestedTreeControl: NestedTreeControl<FileNode>;
|
|
|
|
nestedDataSource: MatTreeNestedDataSource<FileNode>;
|
|
|
|
constructor(database: FileDatabase) {
|
|
this.nestedTreeControl = new NestedTreeControl<FileNode>(this._getChildren);
|
|
this.nestedDataSource = new MatTreeNestedDataSource();
|
|
|
|
database.dataChange.subscribe(data => this.nestedDataSource.data = data);
|
|
}
|
|
|
|
private _getChildren = (node: FileNode) => { return observableOf(node.children); };
|
|
|
|
hasNestedChild = (_: number, nodeData: FileNode) => {return !(nodeData.type); };
|
|
}
|