2015-01-21 12:05:52 -08:00
|
|
|
import {Parser, Lexer, ChangeDetector, ChangeDetection, jitChangeDetection}
|
2015-02-05 13:08:05 -08:00
|
|
|
from 'angular2/change_detection';
|
2015-02-16 14:55:00 +01:00
|
|
|
import {ExceptionHandler} from 'angular2/src/core/exception_handler';
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-02-12 14:44:59 +01:00
|
|
|
import {bootstrap, Component, Viewport, Template, ViewContainer, Compiler} from 'angular2/angular2';
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-02-05 13:08:05 -08:00
|
|
|
import {CompilerCache} from 'angular2/src/core/compiler/compiler';
|
|
|
|
import {DirectiveMetadataReader} from 'angular2/src/core/compiler/directive_metadata_reader';
|
|
|
|
import {TemplateLoader} from 'angular2/src/core/compiler/template_loader';
|
2015-02-12 14:44:59 +01:00
|
|
|
import {TemplateResolver} from 'angular2/src/core/compiler/template_resolver';
|
2015-01-30 09:43:21 +01:00
|
|
|
import {ShadowDomStrategy, NativeShadowDomStrategy} from 'angular2/src/core/compiler/shadow_dom_strategy';
|
2015-02-05 13:08:05 -08:00
|
|
|
import {LifeCycle} from 'angular2/src/core/life_cycle/life_cycle';
|
2015-02-24 16:05:45 +01:00
|
|
|
import {UrlResolver} from 'angular2/src/core/compiler/url_resolver';
|
|
|
|
import {StyleUrlResolver} from 'angular2/src/core/compiler/style_url_resolver';
|
|
|
|
import {ComponentUrlMapper} from 'angular2/src/core/compiler/component_url_mapper';
|
|
|
|
import {StyleInliner} from 'angular2/src/core/compiler/style_inliner';
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-02-05 13:08:05 -08:00
|
|
|
import {reflector} from 'angular2/src/reflection/reflection';
|
|
|
|
import {DOM, document, window, Element, gc} from 'angular2/src/facade/dom';
|
|
|
|
import {isPresent} from 'angular2/src/facade/lang';
|
|
|
|
import {getIntParameter, bindAction} from 'angular2/src/test_lib/benchmark_util';
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-01-30 09:43:21 +01:00
|
|
|
import {XHR} from 'angular2/src/core/compiler/xhr/xhr';
|
|
|
|
import {XHRImpl} from 'angular2/src/core/compiler/xhr/xhr_impl';
|
|
|
|
|
2015-02-20 09:48:16 +01:00
|
|
|
import {If} from 'angular2/directives';
|
|
|
|
|
2014-12-22 17:50:10 -08:00
|
|
|
function setupReflector() {
|
2014-12-08 10:57:09 -08:00
|
|
|
// TODO: Put the general calls to reflector.register... in a shared file
|
|
|
|
// as they are needed in all benchmarks...
|
|
|
|
|
|
|
|
reflector.registerType(AppComponent, {
|
|
|
|
'factory': () => new AppComponent(),
|
|
|
|
'parameters': [],
|
2015-02-12 14:44:59 +01:00
|
|
|
'annotations' : [
|
|
|
|
new Component({selector: 'app'}),
|
|
|
|
new Template({
|
2014-12-08 10:57:09 -08:00
|
|
|
directives: [TreeComponent],
|
|
|
|
inline: `<tree [data]='initData'></tree>`
|
2015-02-12 14:44:59 +01:00
|
|
|
})]
|
2014-12-08 10:57:09 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerType(TreeComponent, {
|
|
|
|
'factory': () => new TreeComponent(),
|
|
|
|
'parameters': [],
|
2015-02-12 14:44:59 +01:00
|
|
|
'annotations' : [
|
|
|
|
new Component({
|
|
|
|
selector: 'tree',
|
|
|
|
bind: {'data': 'data'}
|
|
|
|
}),
|
|
|
|
new Template({
|
2015-02-20 09:48:16 +01:00
|
|
|
directives: [TreeComponent, If],
|
|
|
|
inline: `<span> {{data.value}} <span template='if data.right != null'><tree [data]='data.right'></tree></span><span template='if data.left != null'><tree [data]='data.left'></tree></span></span>`
|
2015-02-12 14:44:59 +01:00
|
|
|
})]
|
2014-12-08 10:57:09 -08:00
|
|
|
});
|
|
|
|
|
2015-02-20 09:48:16 +01:00
|
|
|
reflector.registerType(If, {
|
|
|
|
'factory': (vp) => new If(vp),
|
2015-02-12 11:54:22 +01:00
|
|
|
'parameters': [[ViewContainer]],
|
|
|
|
'annotations' : [new Viewport({
|
2015-02-20 09:48:16 +01:00
|
|
|
selector: '[if]',
|
2014-12-08 10:57:09 -08:00
|
|
|
bind: {
|
2015-02-20 09:48:16 +01:00
|
|
|
'condition': 'if'
|
2014-12-08 10:57:09 -08:00
|
|
|
}
|
|
|
|
})]
|
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerType(Compiler, {
|
2015-02-24 16:05:45 +01:00
|
|
|
'factory': (cd, templateLoader, reader, parser, compilerCache, strategy, tplResolver,
|
|
|
|
cmpUrlMapper, urlResolver) =>
|
|
|
|
new Compiler(cd, templateLoader, reader, parser, compilerCache, strategy, tplResolver,
|
|
|
|
cmpUrlMapper, urlResolver),
|
2015-01-30 09:43:21 +01:00
|
|
|
'parameters': [[ChangeDetection], [TemplateLoader], [DirectiveMetadataReader],
|
2015-02-24 16:05:45 +01:00
|
|
|
[Parser], [CompilerCache], [ShadowDomStrategy], [TemplateResolver],
|
|
|
|
[ComponentUrlMapper], [UrlResolver]],
|
2014-12-08 10:57:09 -08:00
|
|
|
'annotations': []
|
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerType(CompilerCache, {
|
|
|
|
'factory': () => new CompilerCache(),
|
|
|
|
'parameters': [],
|
|
|
|
'annotations': []
|
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerType(Parser, {
|
|
|
|
'factory': (lexer) => new Parser(lexer),
|
|
|
|
'parameters': [[Lexer]],
|
|
|
|
'annotations': []
|
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerType(TemplateLoader, {
|
2015-02-24 16:05:45 +01:00
|
|
|
'factory': (xhr, urlResolver) => new TemplateLoader(xhr, urlResolver),
|
|
|
|
'parameters': [[XHR], [UrlResolver]],
|
2015-01-30 09:43:21 +01:00
|
|
|
'annotations': []
|
|
|
|
});
|
|
|
|
|
2015-02-12 14:44:59 +01:00
|
|
|
reflector.registerType(TemplateResolver, {
|
|
|
|
'factory': () => new TemplateResolver(),
|
|
|
|
'parameters': [],
|
|
|
|
'annotations': []
|
|
|
|
});
|
|
|
|
|
2015-01-30 09:43:21 +01:00
|
|
|
reflector.registerType(XHR, {
|
|
|
|
'factory': () => new XHRImpl(),
|
2014-12-08 10:57:09 -08:00
|
|
|
'parameters': [],
|
|
|
|
'annotations': []
|
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerType(DirectiveMetadataReader, {
|
|
|
|
'factory': () => new DirectiveMetadataReader(),
|
|
|
|
'parameters': [],
|
|
|
|
'annotations': []
|
|
|
|
});
|
|
|
|
|
2015-01-30 09:43:21 +01:00
|
|
|
reflector.registerType(ShadowDomStrategy, {
|
2015-02-24 16:05:45 +01:00
|
|
|
"factory": (strategy) => strategy,
|
|
|
|
"parameters": [[NativeShadowDomStrategy]],
|
|
|
|
"annotations": []
|
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerType(NativeShadowDomStrategy, {
|
|
|
|
"factory": (styleUrlResolver) => new NativeShadowDomStrategy(styleUrlResolver),
|
|
|
|
"parameters": [[StyleUrlResolver]],
|
|
|
|
"annotations": []
|
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerType(StyleUrlResolver, {
|
|
|
|
"factory": (urlResolver) => new StyleUrlResolver(urlResolver),
|
|
|
|
"parameters": [[UrlResolver]],
|
|
|
|
"annotations": []
|
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerType(UrlResolver, {
|
|
|
|
"factory": () => new UrlResolver(),
|
|
|
|
"parameters": [],
|
|
|
|
"annotations": []
|
2015-01-30 09:43:21 +01:00
|
|
|
});
|
|
|
|
|
2014-12-08 10:57:09 -08:00
|
|
|
reflector.registerType(Lexer, {
|
|
|
|
'factory': () => new Lexer(),
|
|
|
|
'parameters': [],
|
|
|
|
'annotations': []
|
|
|
|
});
|
|
|
|
|
2015-02-16 14:55:00 +01:00
|
|
|
reflector.registerType(ExceptionHandler, {
|
|
|
|
"factory": () => new ExceptionHandler(),
|
|
|
|
"parameters": [],
|
|
|
|
"annotations": []
|
|
|
|
});
|
|
|
|
|
2014-12-22 17:50:10 -08:00
|
|
|
reflector.registerType(LifeCycle, {
|
2015-02-16 14:55:00 +01:00
|
|
|
"factory": (exHandler, cd) => new LifeCycle(exHandler, cd),
|
2015-02-25 16:30:30 +01:00
|
|
|
"parameters": [[ExceptionHandler], [ChangeDetector]],
|
2014-12-22 17:50:10 -08:00
|
|
|
"annotations": []
|
|
|
|
});
|
|
|
|
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-02-24 16:05:45 +01:00
|
|
|
reflector.registerType(ComponentUrlMapper, {
|
|
|
|
"factory": () => new ComponentUrlMapper(),
|
|
|
|
"parameters": [],
|
|
|
|
"annotations": []
|
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerType(StyleInliner, {
|
2015-02-26 17:08:37 +01:00
|
|
|
"factory": (xhr, styleUrlResolver, urlResolver) =>
|
|
|
|
new StyleInliner(xhr, styleUrlResolver, urlResolver),
|
|
|
|
"parameters": [[XHR], [StyleUrlResolver], [UrlResolver]],
|
2015-02-24 16:05:45 +01:00
|
|
|
"annotations": []
|
|
|
|
});
|
|
|
|
|
2014-12-08 10:57:09 -08:00
|
|
|
reflector.registerGetters({
|
|
|
|
'value': (a) => a.value,
|
|
|
|
'left': (a) => a.left,
|
|
|
|
'right': (a) => a.right,
|
|
|
|
'initData': (a) => a.initData,
|
2015-02-20 09:48:16 +01:00
|
|
|
'data': (a) => a.data,
|
|
|
|
'condition': (a) => a.condition,
|
2014-12-08 10:57:09 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
reflector.registerSetters({
|
|
|
|
'value': (a,v) => a.value = v,
|
|
|
|
'left': (a,v) => a.left = v,
|
|
|
|
'right': (a,v) => a.right = v,
|
|
|
|
'initData': (a,v) => a.initData = v,
|
|
|
|
'data': (a,v) => a.data = v,
|
2015-02-20 09:48:16 +01:00
|
|
|
'condition': (a,v) => a.condition = v,
|
2014-12-08 10:57:09 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
export function main() {
|
2015-01-09 18:00:04 -08:00
|
|
|
var maxDepth = getIntParameter('depth');
|
|
|
|
|
|
|
|
setupReflector();
|
2014-12-22 17:50:10 -08:00
|
|
|
|
2014-12-08 10:57:09 -08:00
|
|
|
var app;
|
2015-01-14 13:51:16 -08:00
|
|
|
var lifeCycle;
|
2014-12-22 17:50:10 -08:00
|
|
|
var baselineRootTreeComponent;
|
|
|
|
var count = 0;
|
|
|
|
|
2015-01-09 18:00:04 -08:00
|
|
|
function ng2DestroyDom() {
|
2014-12-22 17:50:10 -08:00
|
|
|
// TODO: We need an initial value as otherwise the getter for data.value will fail
|
|
|
|
// --> this should be already caught in change detection!
|
|
|
|
app.initData = new TreeNode('', null, null);
|
2015-01-14 13:51:16 -08:00
|
|
|
lifeCycle.tick();
|
2014-12-22 17:50:10 -08:00
|
|
|
}
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-01-07 21:58:40 -08:00
|
|
|
function profile(create, destroy, name) {
|
2015-01-09 18:00:04 -08:00
|
|
|
return function() {
|
2015-01-13 13:06:09 -08:00
|
|
|
window.console.profile(name + ' w GC');
|
2015-01-07 21:58:40 -08:00
|
|
|
var duration = 0;
|
|
|
|
var count = 0;
|
2015-01-13 13:06:09 -08:00
|
|
|
while(count++ < 150) {
|
|
|
|
gc();
|
2015-01-07 21:58:40 -08:00
|
|
|
var start = window.performance.now();
|
2015-01-09 18:00:04 -08:00
|
|
|
create();
|
2015-01-07 21:58:40 -08:00
|
|
|
duration += window.performance.now() - start;
|
2015-01-09 18:00:04 -08:00
|
|
|
destroy();
|
2015-01-07 21:58:40 -08:00
|
|
|
}
|
2015-01-13 13:06:09 -08:00
|
|
|
window.console.profileEnd(name + ' w GC');
|
|
|
|
window.console.log(`Iterations: ${count}; time: ${duration / count} ms / iteration`);
|
|
|
|
|
|
|
|
window.console.profile(name + ' w/o GC');
|
|
|
|
duration = 0;
|
|
|
|
count = 0;
|
|
|
|
while(count++ < 150) {
|
|
|
|
var start = window.performance.now();
|
2015-01-09 18:00:04 -08:00
|
|
|
create();
|
2015-01-13 13:06:09 -08:00
|
|
|
duration += window.performance.now() - start;
|
2015-01-09 18:00:04 -08:00
|
|
|
destroy();
|
2015-01-13 13:06:09 -08:00
|
|
|
}
|
|
|
|
window.console.profileEnd(name + ' w/o GC');
|
2015-01-07 21:58:40 -08:00
|
|
|
window.console.log(`Iterations: ${count}; time: ${duration / count} ms / iteration`);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2015-01-09 18:00:04 -08:00
|
|
|
function ng2CreateDom() {
|
2014-12-22 17:50:10 -08:00
|
|
|
var values = count++ % 2 == 0 ?
|
|
|
|
['0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '*'] :
|
|
|
|
['A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', '-'];
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-01-09 18:00:04 -08:00
|
|
|
app.initData = buildTree(maxDepth, values, 0);
|
2015-01-14 13:51:16 -08:00
|
|
|
lifeCycle.tick();
|
2014-12-22 17:50:10 -08:00
|
|
|
}
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-01-07 21:58:40 -08:00
|
|
|
function noop() {}
|
|
|
|
|
2014-12-22 17:50:10 -08:00
|
|
|
function initNg2() {
|
|
|
|
bootstrap(AppComponent).then((injector) => {
|
2015-01-14 13:51:16 -08:00
|
|
|
lifeCycle = injector.get(LifeCycle);
|
|
|
|
|
2014-12-22 17:50:10 -08:00
|
|
|
app = injector.get(AppComponent);
|
2015-01-09 18:00:04 -08:00
|
|
|
bindAction('#ng2DestroyDom', ng2DestroyDom);
|
|
|
|
bindAction('#ng2CreateDom', ng2CreateDom);
|
|
|
|
bindAction('#ng2UpdateDomProfile', profile(ng2CreateDom, noop, 'ng2-update'));
|
|
|
|
bindAction('#ng2CreateDomProfile', profile(ng2CreateDom, ng2DestroyDom, 'ng2-create'));
|
2014-12-08 10:57:09 -08:00
|
|
|
});
|
2014-12-22 17:50:10 -08:00
|
|
|
}
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-01-09 18:00:04 -08:00
|
|
|
function baselineDestroyDom() {
|
2014-12-22 17:50:10 -08:00
|
|
|
baselineRootTreeComponent.update(new TreeNode('', null, null));
|
|
|
|
}
|
2014-12-08 14:17:44 -08:00
|
|
|
|
2015-01-09 18:00:04 -08:00
|
|
|
function baselineCreateDom() {
|
2014-12-22 17:50:10 -08:00
|
|
|
var values = count++ % 2 == 0 ?
|
|
|
|
['0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '*'] :
|
|
|
|
['A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', '-'];
|
2014-12-08 14:17:44 -08:00
|
|
|
|
2015-01-09 18:00:04 -08:00
|
|
|
baselineRootTreeComponent.update(buildTree(maxDepth, values, 0));
|
2014-12-22 17:50:10 -08:00
|
|
|
}
|
2014-12-08 14:17:44 -08:00
|
|
|
|
2014-12-22 17:50:10 -08:00
|
|
|
function initBaseline() {
|
2015-01-12 21:56:11 -08:00
|
|
|
var tree = DOM.createElement('tree');
|
|
|
|
DOM.appendChild(DOM.querySelector(document, 'baseline'), tree);
|
|
|
|
baselineRootTreeComponent = new BaseLineTreeComponent(tree);
|
2015-01-09 18:00:04 -08:00
|
|
|
|
|
|
|
bindAction('#baselineDestroyDom', baselineDestroyDom);
|
|
|
|
bindAction('#baselineCreateDom', baselineCreateDom);
|
|
|
|
|
|
|
|
bindAction('#baselineUpdateDomProfile', profile(baselineCreateDom, noop, 'baseline-update'));
|
|
|
|
bindAction('#baselineCreateDomProfile', profile(baselineCreateDom, baselineDestroyDom, 'baseline-create'));
|
2014-12-22 17:50:10 -08:00
|
|
|
}
|
2014-12-08 14:17:44 -08:00
|
|
|
|
2014-12-22 17:50:10 -08:00
|
|
|
initNg2();
|
|
|
|
initBaseline();
|
2014-12-08 10:57:09 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
class TreeNode {
|
|
|
|
value:string;
|
|
|
|
left:TreeNode;
|
|
|
|
right:TreeNode;
|
|
|
|
constructor(value, left, right) {
|
|
|
|
this.value = value;
|
|
|
|
this.left = left;
|
|
|
|
this.right = right;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function buildTree(maxDepth, values, curDepth) {
|
|
|
|
if (maxDepth === curDepth) return new TreeNode('', null, null);
|
|
|
|
return new TreeNode(
|
|
|
|
values[curDepth],
|
|
|
|
buildTree(maxDepth, values, curDepth+1),
|
|
|
|
buildTree(maxDepth, values, curDepth+1));
|
|
|
|
}
|
|
|
|
|
2015-01-12 21:56:11 -08:00
|
|
|
var BASELINE_TREE_TEMPLATE = DOM.createTemplate(
|
2014-12-11 13:58:26 -08:00
|
|
|
'<span>_<template class="ng-binding"></template><template class="ng-binding"></template></span>');
|
2015-01-12 21:56:11 -08:00
|
|
|
var BASELINE_IF_TEMPLATE = DOM.createTemplate(
|
2015-02-20 09:48:16 +01:00
|
|
|
'<span template="if"><tree></tree></span>');
|
2014-12-11 13:58:26 -08:00
|
|
|
// http://jsperf.com/nextsibling-vs-childnodes
|
2014-12-08 14:17:44 -08:00
|
|
|
|
|
|
|
class BaseLineTreeComponent {
|
2015-01-13 13:06:09 -08:00
|
|
|
element:Element;
|
2014-12-08 14:17:44 -08:00
|
|
|
value:BaseLineInterpolation;
|
|
|
|
left:BaseLineIf;
|
|
|
|
right:BaseLineIf;
|
2015-01-12 21:56:11 -08:00
|
|
|
constructor(element) {
|
|
|
|
this.element = element;
|
|
|
|
var clone = DOM.clone(BASELINE_TREE_TEMPLATE.content.firstChild);
|
2014-12-08 14:17:44 -08:00
|
|
|
var shadowRoot = this.element.createShadowRoot();
|
|
|
|
DOM.appendChild(shadowRoot, clone);
|
|
|
|
|
2014-12-11 13:58:26 -08:00
|
|
|
var child = clone.firstChild;
|
|
|
|
this.value = new BaseLineInterpolation(child);
|
|
|
|
child = DOM.nextSibling(child);
|
|
|
|
this.left = new BaseLineIf(child);
|
|
|
|
child = DOM.nextSibling(child);
|
|
|
|
this.right = new BaseLineIf(child);
|
2014-12-08 14:17:44 -08:00
|
|
|
}
|
|
|
|
update(value:TreeNode) {
|
|
|
|
this.value.update(value.value);
|
|
|
|
this.left.update(value.left);
|
|
|
|
this.right.update(value.right);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class BaseLineInterpolation {
|
|
|
|
value:string;
|
|
|
|
textNode;
|
|
|
|
constructor(textNode) {
|
|
|
|
this.value = null;
|
|
|
|
this.textNode = textNode;
|
|
|
|
}
|
|
|
|
update(value:string) {
|
|
|
|
if (this.value !== value) {
|
|
|
|
this.value = value;
|
|
|
|
DOM.setText(this.textNode, value + ' ');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class BaseLineIf {
|
|
|
|
condition:boolean;
|
|
|
|
component:BaseLineTreeComponent;
|
|
|
|
anchor:Element;
|
|
|
|
constructor(anchor) {
|
|
|
|
this.anchor = anchor;
|
|
|
|
this.condition = false;
|
|
|
|
this.component = null;
|
|
|
|
}
|
|
|
|
update(value:TreeNode) {
|
|
|
|
var newCondition = isPresent(value);
|
|
|
|
if (this.condition !== newCondition) {
|
|
|
|
this.condition = newCondition;
|
|
|
|
if (isPresent(this.component)) {
|
2015-02-25 15:56:08 -08:00
|
|
|
DOM.remove(this.component.element);
|
2014-12-08 14:17:44 -08:00
|
|
|
this.component = null;
|
|
|
|
}
|
|
|
|
if (this.condition) {
|
2015-01-13 11:14:05 -08:00
|
|
|
var element = DOM.firstChild(DOM.clone(BASELINE_IF_TEMPLATE).content);
|
|
|
|
this.anchor.parentNode.insertBefore(element, DOM.nextSibling(this.anchor));
|
|
|
|
this.component = new BaseLineTreeComponent(DOM.firstChild(element));
|
2014-12-08 14:17:44 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (isPresent(this.component)) {
|
|
|
|
this.component.update(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-08 10:57:09 -08:00
|
|
|
class AppComponent {
|
|
|
|
initData:TreeNode;
|
|
|
|
constructor() {
|
|
|
|
// TODO: We need an initial value as otherwise the getter for data.value will fail
|
|
|
|
// --> this should be already caught in change detection!
|
|
|
|
this.initData = new TreeNode('', null, null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class TreeComponent {
|
|
|
|
data:TreeNode;
|
|
|
|
}
|
|
|
|
|