2020-07-10 10:02:08 +00:00
|
|
|
// Copyright 2020 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
2020-08-13 12:55:19 +00:00
|
|
|
import { V8CustomElement, defineCustomElement } from "./helper.mjs";
|
|
|
|
import { SelectionEvent } from "./events.mjs";
|
2020-07-10 10:02:08 +00:00
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
defineCustomElement(
|
|
|
|
"stats-panel",
|
|
|
|
(templateText) =>
|
|
|
|
class StatsPanel extends V8CustomElement {
|
|
|
|
#timeline;
|
|
|
|
#transitions;
|
|
|
|
constructor() {
|
|
|
|
super(templateText);
|
|
|
|
}
|
2020-07-10 10:02:08 +00:00
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
get stats() {
|
|
|
|
return this.$("#stats");
|
|
|
|
}
|
2020-07-10 10:02:08 +00:00
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
set timeline(value) {
|
|
|
|
//TODO(zcankara) Trigger update
|
|
|
|
this.#timeline = value;
|
|
|
|
}
|
2020-07-10 10:02:08 +00:00
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
get timeline() {
|
|
|
|
return this.#timeline;
|
|
|
|
}
|
2020-07-29 11:41:42 +00:00
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
set transitions(value) {
|
|
|
|
this.#transitions = value;
|
|
|
|
}
|
2020-07-24 13:51:36 +00:00
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
get transitions() {
|
|
|
|
return this.#transitions;
|
|
|
|
}
|
2020-07-24 13:51:36 +00:00
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
filterUniqueTransitions(filter) {
|
|
|
|
// Returns a list of Maps whose parent is not in the list.
|
|
|
|
return this.timeline.filter((map) => {
|
|
|
|
if (filter(map) === false) return false;
|
|
|
|
let parent = map.parent();
|
|
|
|
if (parent === undefined) return true;
|
|
|
|
return filter(parent) === false;
|
|
|
|
});
|
|
|
|
}
|
2020-07-10 10:02:08 +00:00
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
update() {
|
|
|
|
this.removeAllChildren(this.stats);
|
|
|
|
this.updateGeneralStats();
|
|
|
|
this.updateNamedTransitionsStats();
|
|
|
|
}
|
2020-07-10 10:02:08 +00:00
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
updateGeneralStats() {
|
|
|
|
console.assert(this.#timeline !== undefined, "Timeline not set yet!");
|
|
|
|
let pairs = [
|
|
|
|
["Total", null, (e) => true],
|
|
|
|
["Transitions", "primary", (e) => e.edge && e.edge.isTransition()],
|
|
|
|
["Fast to Slow", "violet", (e) => e.edge && e.edge.isFastToSlow()],
|
|
|
|
["Slow to Fast", "orange", (e) => e.edge && e.edge.isSlowToFast()],
|
|
|
|
["Initial Map", "yellow", (e) => e.edge && e.edge.isInitial()],
|
|
|
|
[
|
|
|
|
"Replace Descriptors",
|
|
|
|
"red",
|
|
|
|
(e) => e.edge && e.edge.isReplaceDescriptors(),
|
|
|
|
],
|
|
|
|
[
|
|
|
|
"Copy as Prototype",
|
|
|
|
"red",
|
|
|
|
(e) => e.edge && e.edge.isCopyAsPrototype(),
|
|
|
|
],
|
|
|
|
[
|
|
|
|
"Optimize as Prototype",
|
|
|
|
null,
|
|
|
|
(e) => e.edge && e.edge.isOptimizeAsPrototype(),
|
|
|
|
],
|
|
|
|
["Deprecated", null, (e) => e.isDeprecated()],
|
|
|
|
["Bootstrapped", "green", (e) => e.isBootstrapped()],
|
|
|
|
];
|
2020-07-10 10:02:08 +00:00
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
let text = "";
|
|
|
|
let tableNode = this.table("transitionType");
|
|
|
|
tableNode.innerHTML =
|
2020-08-25 06:31:43 +00:00
|
|
|
"<thead><tr><td>Color</td><td>Type</td><td>Count</td>" +
|
|
|
|
"<td>Percent</td></tr></thead>";
|
2020-08-13 12:55:19 +00:00
|
|
|
let name, filter;
|
|
|
|
let total = this.timeline.size();
|
|
|
|
pairs.forEach(([name, color, filter]) => {
|
|
|
|
let row = this.tr();
|
|
|
|
if (color !== null) {
|
|
|
|
row.appendChild(this.td(this.div(["colorbox", color])));
|
|
|
|
} else {
|
|
|
|
row.appendChild(this.td(""));
|
|
|
|
}
|
2020-08-24 06:21:26 +00:00
|
|
|
row.classList.add('clickable');
|
2020-08-13 12:55:19 +00:00
|
|
|
row.onclick = (e) => {
|
|
|
|
// lazily compute the stats
|
|
|
|
let node = e.target.parentNode;
|
|
|
|
if (node.maps == undefined) {
|
|
|
|
node.maps = this.filterUniqueTransitions(filter);
|
|
|
|
}
|
|
|
|
this.dispatchEvent(new SelectionEvent(node.maps));
|
|
|
|
};
|
|
|
|
row.appendChild(this.td(name));
|
|
|
|
let count = this.timeline.count(filter);
|
|
|
|
row.appendChild(this.td(count));
|
|
|
|
let percent = Math.round((count / total) * 1000) / 10;
|
|
|
|
row.appendChild(this.td(percent.toFixed(1) + "%"));
|
|
|
|
tableNode.appendChild(row);
|
|
|
|
});
|
|
|
|
this.stats.appendChild(tableNode);
|
2020-07-10 10:02:08 +00:00
|
|
|
}
|
|
|
|
|
2020-08-13 12:55:19 +00:00
|
|
|
updateNamedTransitionsStats() {
|
|
|
|
let tableNode = this.table("transitionTable");
|
|
|
|
let nameMapPairs = Array.from(this.transitions.entries());
|
|
|
|
tableNode.innerHTML =
|
|
|
|
"<thead><tr><td>Propery Name</td><td>#</td></tr></thead>";
|
|
|
|
nameMapPairs
|
|
|
|
.sort((a, b) => b[1].length - a[1].length)
|
|
|
|
.forEach(([name, maps]) => {
|
|
|
|
let row = this.tr();
|
|
|
|
row.maps = maps;
|
2020-08-24 06:21:26 +00:00
|
|
|
row.classList.add('clickable');
|
2020-08-13 12:55:19 +00:00
|
|
|
row.addEventListener("click", (e) =>
|
|
|
|
this.dispatchEvent(
|
|
|
|
new SelectionEvent(
|
|
|
|
e.target.parentNode.maps.map((map) => map.to)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
row.appendChild(this.td(name));
|
|
|
|
row.appendChild(this.td(maps.length));
|
|
|
|
tableNode.appendChild(row);
|
|
|
|
});
|
|
|
|
this.stats.appendChild(tableNode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|