2020-06-29 19:21:56 +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-07-16 14:46:13 +00:00
|
|
|
import {defineCustomElement, V8CustomElement} from './helper.mjs';
|
2020-06-29 19:21:56 +00:00
|
|
|
|
|
|
|
defineCustomElement('timeline-panel', (templateText) =>
|
2020-07-16 14:46:13 +00:00
|
|
|
class TimelinePanel extends V8CustomElement {
|
2020-06-29 19:21:56 +00:00
|
|
|
constructor() {
|
2020-07-16 14:46:13 +00:00
|
|
|
super(templateText);
|
2020-07-15 04:51:13 +00:00
|
|
|
this.timelineOverview.addEventListener(
|
2020-06-29 19:21:56 +00:00
|
|
|
'mousemove', e => this.handleTimelineIndicatorMove(e));
|
|
|
|
}
|
|
|
|
|
2020-07-15 04:51:13 +00:00
|
|
|
get timelineOverview() {
|
2020-06-29 19:21:56 +00:00
|
|
|
return this.$('#timelineOverview');
|
|
|
|
}
|
|
|
|
|
2020-07-15 04:51:13 +00:00
|
|
|
get timelineOverviewIndicator() {
|
2020-06-29 19:21:56 +00:00
|
|
|
return this.$('#timelineOverviewIndicator');
|
|
|
|
}
|
|
|
|
|
2020-07-15 04:51:13 +00:00
|
|
|
get timelineCanvas() {
|
2020-06-29 19:21:56 +00:00
|
|
|
return this.$('#timelineCanvas');
|
|
|
|
}
|
|
|
|
|
2020-07-15 04:51:13 +00:00
|
|
|
get timelineChunks() {
|
2020-06-29 19:21:56 +00:00
|
|
|
return this.$('#timelineChunks');
|
|
|
|
}
|
|
|
|
|
2020-07-15 04:51:13 +00:00
|
|
|
get timeline() {
|
2020-06-29 19:21:56 +00:00
|
|
|
return this.$('#timeline');
|
|
|
|
}
|
|
|
|
|
2020-07-16 19:24:09 +00:00
|
|
|
// Timeline related View component methods
|
|
|
|
createBackgroundCanvas(){
|
|
|
|
this.backgroundCanvas = document.createElement('canvas');
|
|
|
|
}
|
|
|
|
|
2020-06-29 19:21:56 +00:00
|
|
|
handleTimelineIndicatorMove(event) {
|
|
|
|
if (event.buttons == 0) return;
|
2020-07-15 04:51:13 +00:00
|
|
|
let timelineTotalWidth = this.timelineCanvas.offsetWidth;
|
|
|
|
let factor = this.timelineOverview.offsetWidth / timelineTotalWidth;
|
|
|
|
this.timeline.scrollLeft += event.movementX / factor;
|
2020-06-29 19:21:56 +00:00
|
|
|
}
|
|
|
|
|
2020-07-16 19:24:09 +00:00
|
|
|
updateOverviewWindow() {
|
|
|
|
let indicator = this.timelineOverviewIndicator;
|
|
|
|
let totalIndicatorWidth =
|
|
|
|
this.timelineOverview.offsetWidth;
|
|
|
|
let div = this.timeline;
|
|
|
|
let timelineTotalWidth =
|
|
|
|
this.timelineCanvas.offsetWidth;
|
|
|
|
let factor = totalIndicatorWidth / timelineTotalWidth;
|
|
|
|
let width = div.offsetWidth * factor;
|
|
|
|
let left = div.scrollLeft * factor;
|
|
|
|
indicator.style.width = width + 'px';
|
|
|
|
indicator.style.left = left + 'px';
|
|
|
|
}
|
|
|
|
|
|
|
|
asyncSetTimelineChunkBackground(backgroundTodo) {
|
|
|
|
const kIncrement = 100;
|
|
|
|
let start = 0;
|
|
|
|
let delay = 1;
|
|
|
|
while (start < backgroundTodo.length) {
|
|
|
|
let end = Math.min(start + kIncrement, backgroundTodo.length);
|
|
|
|
setTimeout((from, to) => {
|
|
|
|
for (let i = from; i < to; i++) {
|
|
|
|
let [chunk, node] = backgroundTodo[i];
|
|
|
|
this.setTimelineChunkBackground(chunk, node);
|
|
|
|
}
|
|
|
|
}, delay++, start, end);
|
|
|
|
start = end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setTimelineChunkBackground(chunk, node) {
|
|
|
|
// Render the types of transitions as bar charts
|
|
|
|
const kHeight = chunk.height;
|
|
|
|
const kWidth = 1;
|
|
|
|
this.backgroundCanvas.width = kWidth;
|
|
|
|
this.backgroundCanvas.height = kHeight;
|
|
|
|
let ctx = this.backgroundCanvas.getContext('2d');
|
|
|
|
ctx.clearRect(0, 0, kWidth, kHeight);
|
|
|
|
let y = 0;
|
|
|
|
let total = chunk.size();
|
|
|
|
let type, count;
|
|
|
|
if (true) {
|
|
|
|
chunk.getTransitionBreakdown().forEach(([type, count]) => {
|
|
|
|
ctx.fillStyle = this.transitionTypeToColor(type);
|
|
|
|
let height = count / total * kHeight;
|
|
|
|
ctx.fillRect(0, y, kWidth, y + height);
|
|
|
|
y += height;
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
chunk.items.forEach(map => {
|
|
|
|
ctx.fillStyle = this.transitionTypeToColor(map.getType());
|
|
|
|
let y = chunk.yOffset(map);
|
|
|
|
ctx.fillRect(0, y, kWidth, y + 1);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
let imageData = this.backgroundCanvas.toDataURL('image/webp', 0.2);
|
|
|
|
node.style.backgroundImage = 'url(' + imageData + ')';
|
|
|
|
}
|
|
|
|
|
|
|
|
transitionTypeToColor(type) {
|
|
|
|
switch (type) {
|
|
|
|
case 'new':
|
|
|
|
// green
|
|
|
|
return '#aedc6e';
|
|
|
|
case 'Normalize':
|
|
|
|
// violet
|
|
|
|
return '#d26edc';
|
|
|
|
case 'SlowToFast':
|
|
|
|
// orange
|
|
|
|
return '#dc9b6e';
|
|
|
|
case 'InitialMap':
|
|
|
|
// yellow
|
|
|
|
return '#EEFF41';
|
|
|
|
case 'Transition':
|
|
|
|
// pink/violet (primary)
|
|
|
|
return '#9B6EDC';
|
|
|
|
case 'ReplaceDescriptors':
|
|
|
|
// red
|
|
|
|
return '#dc6eae';
|
|
|
|
}
|
|
|
|
// pink/violet (primary)
|
|
|
|
return '#9B6EDC';
|
|
|
|
}
|
|
|
|
|
2020-06-29 19:21:56 +00:00
|
|
|
});
|