2015-09-03 22:01:36 -07:00
|
|
|
import {bootstrap} from 'angular2/bootstrap';
|
2015-07-22 10:18:04 -07:00
|
|
|
import {
|
|
|
|
Compiler,
|
|
|
|
Component,
|
|
|
|
Directive,
|
|
|
|
View,
|
2015-09-03 22:01:36 -07:00
|
|
|
ViewContainerRef,
|
|
|
|
bind,
|
2015-10-10 22:11:13 -07:00
|
|
|
provide,
|
2015-11-18 15:55:43 -08:00
|
|
|
Provider
|
2015-09-03 22:01:36 -07:00
|
|
|
} from 'angular2/core';
|
2015-11-18 15:55:43 -08:00
|
|
|
import {NgIf} from 'angular2/common';
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-10-28 10:34:13 -07:00
|
|
|
import {ApplicationRef} from 'angular2/src/core/application_ref';
|
2015-11-19 15:09:34 -08:00
|
|
|
import {DOM} from 'angular2/src/platform/dom/dom_adapter';
|
2015-11-06 17:34:07 -08:00
|
|
|
import {isPresent} from 'angular2/src/facade/lang';
|
|
|
|
import {window, document, gc} from 'angular2/src/facade/browser';
|
2015-05-29 21:40:15 -07:00
|
|
|
import {
|
|
|
|
getIntParameter,
|
|
|
|
getStringParameter,
|
2015-05-30 08:17:27 -07:00
|
|
|
bindAction,
|
|
|
|
windowProfile,
|
|
|
|
windowProfileEnd
|
2015-10-13 00:29:13 -07:00
|
|
|
} from 'angular2/src/testing/benchmark_util';
|
2015-11-17 15:24:36 -08:00
|
|
|
import {BrowserDomAdapter} from 'angular2/src/platform/browser/browser_adapter';
|
2015-10-02 07:37:23 -07:00
|
|
|
import {APP_VIEW_POOL_CAPACITY} from 'angular2/src/core/linker/view_pool';
|
2015-04-09 17:53:36 -07:00
|
|
|
|
2015-10-10 22:11:13 -07:00
|
|
|
function createProviders(): Provider[] {
|
2015-04-10 13:23:34 -07:00
|
|
|
var viewCacheCapacity = getStringParameter('viewcache') == 'true' ? 10000 : 1;
|
2015-10-12 11:30:34 -07:00
|
|
|
return [provide(APP_VIEW_POOL_CAPACITY, {useValue: viewCacheCapacity})];
|
2015-04-09 17:53:36 -07:00
|
|
|
}
|
2015-03-24 10:19:05 +01:00
|
|
|
|
2015-03-02 15:48:18 -08:00
|
|
|
var BASELINE_TREE_TEMPLATE;
|
|
|
|
var BASELINE_IF_TEMPLATE;
|
|
|
|
|
2014-12-08 10:57:09 -08:00
|
|
|
export function main() {
|
2015-02-27 14:50:06 -08:00
|
|
|
BrowserDomAdapter.makeCurrent();
|
|
|
|
var maxDepth = getIntParameter('depth');
|
2015-01-09 18:00:04 -08:00
|
|
|
|
2015-03-02 15:48:18 -08:00
|
|
|
BASELINE_TREE_TEMPLATE = DOM.createTemplate(
|
2015-10-10 22:11:13 -07:00
|
|
|
'<span>_<template class="ng-provider"></template><template class="ng-provider"></template></span>');
|
2015-05-29 21:40:15 -07:00
|
|
|
BASELINE_IF_TEMPLATE = DOM.createTemplate('<span template="if"><tree></tree></span>');
|
2015-03-02 15:48:18 -08:00
|
|
|
|
2014-12-08 10:57:09 -08:00
|
|
|
var app;
|
2015-10-28 10:34:13 -07:00
|
|
|
var appRef;
|
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-10-28 10:34:13 -07:00
|
|
|
appRef.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-05-30 08:17:27 -07:00
|
|
|
windowProfile(name + ' w GC');
|
2015-01-07 21:58:40 -08:00
|
|
|
var duration = 0;
|
|
|
|
var count = 0;
|
2015-05-29 21:40:15 -07:00
|
|
|
while (count++ < 150) {
|
2015-01-13 13:06:09 -08:00
|
|
|
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-05-30 08:17:27 -07:00
|
|
|
windowProfileEnd(name + ' w GC');
|
2015-01-13 13:06:09 -08:00
|
|
|
window.console.log(`Iterations: ${count}; time: ${duration / count} ms / iteration`);
|
|
|
|
|
2015-05-30 08:17:27 -07:00
|
|
|
windowProfile(name + ' w/o GC');
|
2015-01-13 13:06:09 -08:00
|
|
|
duration = 0;
|
|
|
|
count = 0;
|
2015-05-29 21:40:15 -07:00
|
|
|
while (count++ < 150) {
|
2015-01-13 13:06:09 -08:00
|
|
|
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
|
|
|
}
|
2015-05-30 08:17:27 -07:00
|
|
|
windowProfileEnd(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() {
|
2015-05-29 21:40:15 -07: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', '-'];
|
2015-01-09 18:00:04 -08:00
|
|
|
app.initData = buildTree(maxDepth, values, 0);
|
2015-10-28 10:34:13 -07:00
|
|
|
appRef.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() {
|
2015-10-10 22:11:13 -07:00
|
|
|
bootstrap(AppComponent, createProviders())
|
2015-05-29 21:40:15 -07:00
|
|
|
.then((ref) => {
|
2015-10-08 09:57:10 -07:00
|
|
|
var injector = ref.injector;
|
2015-10-28 10:34:13 -07:00
|
|
|
appRef = injector.get(ApplicationRef);
|
2015-05-29 21:40:15 -07:00
|
|
|
|
2015-06-26 15:59:18 -07:00
|
|
|
app = ref.hostComponent;
|
2015-05-29 21:40:15 -07:00
|
|
|
bindAction('#ng2DestroyDom', ng2DestroyDom);
|
|
|
|
bindAction('#ng2CreateDom', ng2CreateDom);
|
|
|
|
bindAction('#ng2UpdateDomProfile', profile(ng2CreateDom, noop, 'ng2-update'));
|
|
|
|
bindAction('#ng2CreateDomProfile', profile(ng2CreateDom, ng2DestroyDom, 'ng2-create'));
|
|
|
|
});
|
2014-12-22 17:50:10 -08:00
|
|
|
}
|
2014-12-08 10:57:09 -08:00
|
|
|
|
2015-05-29 21:40:15 -07:00
|
|
|
function baselineDestroyDom() { baselineRootTreeComponent.update(new TreeNode('', null, null)); }
|
2014-12-08 14:17:44 -08:00
|
|
|
|
2015-01-09 18:00:04 -08:00
|
|
|
function baselineCreateDom() {
|
2015-05-29 21:40:15 -07: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'));
|
2015-05-29 21:40:15 -07:00
|
|
|
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 {
|
2015-05-29 21:40:15 -07:00
|
|
|
value: string;
|
|
|
|
left: TreeNode;
|
|
|
|
right: TreeNode;
|
2014-12-08 10:57:09 -08:00
|
|
|
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);
|
2015-05-29 21:40:15 -07:00
|
|
|
return new TreeNode(values[curDepth], buildTree(maxDepth, values, curDepth + 1),
|
|
|
|
buildTree(maxDepth, values, curDepth + 1));
|
2014-12-08 10:57:09 -08:00
|
|
|
}
|
|
|
|
|
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-02-27 14:50:06 -08:00
|
|
|
element;
|
2015-05-29 21:40:15 -07: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
|
|
|
}
|
2015-05-29 21:40:15 -07:00
|
|
|
update(value: TreeNode) {
|
2014-12-08 14:17:44 -08:00
|
|
|
this.value.update(value.value);
|
|
|
|
this.left.update(value.left);
|
|
|
|
this.right.update(value.right);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class BaseLineInterpolation {
|
2015-05-29 21:40:15 -07:00
|
|
|
value: string;
|
2014-12-08 14:17:44 -08:00
|
|
|
textNode;
|
|
|
|
constructor(textNode) {
|
|
|
|
this.value = null;
|
|
|
|
this.textNode = textNode;
|
|
|
|
}
|
2015-05-29 21:40:15 -07:00
|
|
|
update(value: string) {
|
2014-12-08 14:17:44 -08:00
|
|
|
if (this.value !== value) {
|
|
|
|
this.value = value;
|
|
|
|
DOM.setText(this.textNode, value + ' ');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class BaseLineIf {
|
2015-05-29 21:40:15 -07:00
|
|
|
condition: boolean;
|
|
|
|
component: BaseLineTreeComponent;
|
2015-02-27 14:50:06 -08:00
|
|
|
anchor;
|
2014-12-08 14:17:44 -08:00
|
|
|
constructor(anchor) {
|
|
|
|
this.anchor = anchor;
|
|
|
|
this.condition = false;
|
|
|
|
this.component = null;
|
|
|
|
}
|
2015-05-29 21:40:15 -07:00
|
|
|
update(value: TreeNode) {
|
2014-12-08 14:17:44 -08:00
|
|
|
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-08-06 09:52:33 -07:00
|
|
|
var element = DOM.firstChild((<any>DOM.clone(BASELINE_IF_TEMPLATE)).content);
|
2015-01-13 11:14:05 -08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-30 20:59:23 -07:00
|
|
|
@Component({selector: 'tree', inputs: ['data']})
|
2015-05-30 08:17:27 -07:00
|
|
|
@View({
|
|
|
|
directives: [TreeComponent, NgIf],
|
|
|
|
template:
|
|
|
|
`<span> {{data.value}} <span template='ng-if data.right != null'><tree [data]='data.right'></tree></span><span template='ng-if data.left != null'><tree [data]='data.left'></tree></span></span>`
|
|
|
|
})
|
|
|
|
class TreeComponent {
|
|
|
|
data: TreeNode;
|
|
|
|
}
|
|
|
|
|
2015-04-09 17:53:36 -07:00
|
|
|
@Component({selector: 'app'})
|
2015-05-29 21:40:15 -07:00
|
|
|
@View({directives: [TreeComponent], template: `<tree [data]='initData'></tree>`})
|
2014-12-08 10:57:09 -08:00
|
|
|
class AppComponent {
|
2015-05-29 21:40:15 -07:00
|
|
|
initData: TreeNode;
|
2014-12-08 10:57:09 -08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|