blob: 23507ee707ebc2ab7220fdc49db1087c967af85b [file] [log] [blame]
// Copyright (C) 2018 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import * as m from 'mithril';
import {assertExists} from '../base/logging';
import {hueForCpu} from '../common/colorizer';
import {TimeSpan, timeToString} from '../common/time';
import {TRACK_SHELL_WIDTH} from './css_constants';
import {DragGestureHandler} from './drag_gesture_handler';
import {globals} from './globals';
import {Panel, PanelSize} from './panel';
import {TimeScale} from './time_scale';
export class OverviewTimelinePanel extends Panel {
private width = 0;
private dragStartPx = 0;
private gesture?: DragGestureHandler;
private timeScale?: TimeScale;
private totTime = new TimeSpan(0, 0);
// Must explicitly type now; arguments types are no longer auto-inferred.
// https://github.com/Microsoft/TypeScript/issues/1373
onupdate({dom}: m.CVnodeDOM) {
this.width = dom.getBoundingClientRect().width;
this.totTime = new TimeSpan(
globals.state.traceTime.startSec, globals.state.traceTime.endSec);
this.timeScale = new TimeScale(
this.totTime, [TRACK_SHELL_WIDTH, assertExists(this.width)]);
if (this.gesture === undefined) {
this.gesture = new DragGestureHandler(
dom as HTMLElement,
this.onDrag.bind(this),
this.onDragStart.bind(this),
this.onDragEnd.bind(this));
}
}
oncreate(vnode: m.CVnodeDOM) {
this.onupdate(vnode);
}
view() {
return m('.overview-timeline');
}
renderCanvas(ctx: CanvasRenderingContext2D, size: PanelSize) {
if (this.width === undefined) return;
if (this.timeScale === undefined) return;
const headerHeight = 25;
const tracksHeight = size.height - headerHeight;
// Draw time labels on the top header.
ctx.font = '10px Roboto Condensed';
ctx.fillStyle = '#999';
for (let i = 0; i < 100; i++) {
const xPos =
(i * (this.width - TRACK_SHELL_WIDTH) / 100) + TRACK_SHELL_WIDTH;
const t = this.timeScale.pxToTime(xPos);
if (xPos <= 0) continue;
if (xPos > this.width) break;
if (i % 10 === 0) {
ctx.fillRect(xPos - 1, 0, 1, headerHeight - 5);
ctx.fillText(timeToString(t - this.totTime.start), xPos + 5, 18);
} else {
ctx.fillRect(xPos - 1, 0, 1, 5);
}
}
// Draw mini-tracks with quanitzed density for each process.
if (globals.overviewStore.size > 0) {
const numTracks = globals.overviewStore.size;
let y = 0;
const trackHeight = (tracksHeight - 1) / numTracks;
for (const key of globals.overviewStore.keys()) {
const loads = globals.overviewStore.get(key)!;
for (let i = 0; i < loads.length; i++) {
const xStart = Math.floor(this.timeScale.timeToPx(loads[i].startSec));
const xEnd = Math.ceil(this.timeScale.timeToPx(loads[i].endSec));
const yOff = Math.floor(headerHeight + y * trackHeight);
const lightness = Math.ceil((1 - loads[i].load * 0.7) * 100);
ctx.fillStyle = `hsl(${hueForCpu(y)}, 50%, ${lightness}%)`;
ctx.fillRect(xStart, yOff, xEnd - xStart, Math.ceil(trackHeight));
}
y++;
}
}
// Draw bottom border.
ctx.fillStyle = '#dadada';
ctx.fillRect(0, size.height - 1, this.width, 1);
// Draw semi-opaque rects that occlude the non-visible time range.
const vizTime = globals.frontendLocalState.visibleWindowTime;
const vizStartPx = Math.floor(this.timeScale.timeToPx(vizTime.start));
const vizEndPx = Math.ceil(this.timeScale.timeToPx(vizTime.end));
ctx.fillStyle = 'rgba(200, 200, 200, 0.8)';
ctx.fillRect(
TRACK_SHELL_WIDTH - 1,
headerHeight,
vizStartPx - TRACK_SHELL_WIDTH,
tracksHeight);
ctx.fillRect(vizEndPx, headerHeight, this.width - vizEndPx, tracksHeight);
// Draw brushes.
ctx.fillStyle = '#999';
ctx.fillRect(vizStartPx - 1, headerHeight, 1, tracksHeight);
ctx.fillRect(vizEndPx, headerHeight, 1, tracksHeight);
}
onDrag(x: number) {
// Set visible time limits from selection.
if (this.timeScale === undefined) return;
let tStart = this.timeScale.pxToTime(this.dragStartPx);
let tEnd = this.timeScale.pxToTime(x);
if (tStart > tEnd) [tStart, tEnd] = [tEnd, tStart];
const vizTime = new TimeSpan(tStart, tEnd);
globals.frontendLocalState.updateVisibleTime(vizTime);
globals.rafScheduler.scheduleRedraw();
}
onDragStart(x: number) {
this.dragStartPx = x;
}
onDragEnd() {
this.dragStartPx = 0;
}
}