blob: 3fcfe3e8b369cd7fa83f85a33c3585bf73195f4d [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 {searchSegment} from '../../base/binary_search';
import {assertTrue} from '../../base/logging';
import {Actions} from '../../common/actions';
import {TrackState} from '../../common/state';
import {toNs} from '../../common/time';
import {checkerboardExcept} from '../../frontend/checkerboard';
import {globals} from '../../frontend/globals';
import {Track} from '../../frontend/track';
import {TrackButton, TrackButtonAttrs} from '../../frontend/track_panel';
import {trackRegistry} from '../../frontend/track_registry';
import {
Config,
COUNTER_TRACK_KIND,
Data,
} from './common';
// 0.5 Makes the horizontal lines sharp.
const MARGIN_TOP = 3.5;
const RECT_HEIGHT = 24.5;
class CounterTrack extends Track<Config, Data> {
static readonly kind = COUNTER_TRACK_KIND;
static create(trackState: TrackState): CounterTrack {
return new CounterTrack(trackState);
}
private mouseXpos = 0;
private hoveredValue: number|undefined = undefined;
private hoveredTs: number|undefined = undefined;
private hoveredTsEnd: number|undefined = undefined;
constructor(trackState: TrackState) {
super(trackState);
}
getHeight() {
return MARGIN_TOP + RECT_HEIGHT;
}
getTrackShellButtons(): Array<m.Vnode<TrackButtonAttrs>> {
const buttons: Array<m.Vnode<TrackButtonAttrs>> = [];
buttons.push(m(TrackButton, {
action: () => {
if (this.config.scale === 'RELATIVE') {
this.config.scale = 'DEFAULT';
} else {
this.config.scale = 'RELATIVE';
}
Actions.updateTrackConfig(
{id: this.trackState.id, config: this.config});
globals.rafScheduler.scheduleFullRedraw();
},
i: 'show_chart',
tooltip: (this.config.scale === 'RELATIVE') ? 'Use zero-based scale' :
'Use relative scale',
selected: this.config.scale === 'RELATIVE',
}));
return buttons;
}
renderCanvas(ctx: CanvasRenderingContext2D): void {
// TODO: fonts and colors should come from the CSS and not hardcoded here.
const {timeScale, visibleWindowTime} = globals.frontendLocalState;
const data = this.data();
if (data === undefined) return; // Can't possibly draw anything.
assertTrue(data.timestamps.length === data.values.length);
const endPx = Math.floor(timeScale.timeToPx(visibleWindowTime.end));
const zeroY = MARGIN_TOP + RECT_HEIGHT / (data.minimumValue < 0 ? 2 : 1);
let lastX = Math.floor(timeScale.timeToPx(data.timestamps[0]));
let lastY = zeroY;
// Quantize the Y axis to quarters of powers of tens (7.5K, 10K, 12.5K).
const maxValue = Math.max(data.maximumValue, 0);
let yMax = Math.max(Math.abs(data.minimumValue), maxValue);
const kUnits = ['', 'K', 'M', 'G', 'T', 'E'];
const exp = Math.ceil(Math.log10(Math.max(yMax, 1)));
const pow10 = Math.pow(10, exp);
yMax = Math.ceil(yMax / (pow10 / 4)) * (pow10 / 4);
let yRange = 0;
const unitGroup = Math.floor(exp / 3);
let yMin = 0;
let yLabel = '';
if (this.config.scale === 'RELATIVE') {
yRange = data.maximumValue - data.minimumValue;
yMin = data.minimumValue;
yLabel = 'min - max';
} else {
yRange = data.minimumValue < 0 ? yMax * 2 : yMax;
yMin = data.minimumValue < 0 ? -yMax : 0;
yLabel = `${yMax / Math.pow(10, unitGroup * 3)} ${kUnits[unitGroup]}`;
}
// There are 360deg of hue. We want a scale that starts at green with
// exp <= 3 (<= 1KB), goes orange around exp = 6 (~1MB) and red/violet
// around exp >= 9 (1GB).
// The hue scale looks like this:
// 0 180 360
// Red orange green | blue purple magenta
// So we want to start @ 180deg with pow=0, go down to 0deg and then wrap
// back from 360deg back to 180deg.
const expCapped = Math.min(Math.max(exp - 3), 9);
const hue = (180 - Math.floor(expCapped * (180 / 6)) + 360) % 360;
ctx.fillStyle = `hsl(${hue}, 45%, 75%)`;
ctx.strokeStyle = `hsl(${hue}, 45%, 45%)`;
ctx.beginPath();
ctx.moveTo(lastX, lastY);
for (let i = 0; i < data.values.length; i++) {
const value = data.values[i];
const startTime = data.timestamps[i];
const nextY = zeroY - Math.round(((value - yMin) / yRange) * RECT_HEIGHT);
if (nextY === lastY) continue;
lastX = Math.floor(timeScale.timeToPx(startTime));
ctx.lineTo(lastX, lastY);
ctx.lineTo(lastX, nextY);
lastY = nextY;
}
ctx.lineTo(endPx, lastY);
ctx.lineTo(endPx, zeroY);
ctx.closePath();
ctx.fill();
ctx.stroke();
// Draw the Y=0 dashed line.
ctx.strokeStyle = `hsl(${hue}, 10%, 71%)`;
ctx.beginPath();
ctx.setLineDash([2, 4]);
ctx.moveTo(0, zeroY);
ctx.lineTo(endPx, zeroY);
ctx.closePath();
ctx.stroke();
ctx.setLineDash([]);
ctx.font = '10px Google Sans';
if (this.hoveredValue !== undefined && this.hoveredTs !== undefined) {
// TODO(hjd): Add units.
let text = (data.isQuantized) ? 'max value: ' : 'value: ';
text += `${this.hoveredValue.toLocaleString()}`;
const width = ctx.measureText(text).width;
ctx.fillStyle = `hsl(${hue}, 45%, 75%)`;
ctx.strokeStyle = `hsl(${hue}, 45%, 45%)`;
const xStart = Math.floor(timeScale.timeToPx(this.hoveredTs));
const xEnd = this.hoveredTsEnd === undefined ?
endPx :
Math.floor(timeScale.timeToPx(this.hoveredTsEnd));
const y = zeroY -
Math.round(((this.hoveredValue - yMin) / yRange) * RECT_HEIGHT);
// Highlight line.
ctx.beginPath();
ctx.moveTo(xStart, y);
ctx.lineTo(xEnd, y);
ctx.lineWidth = 3;
ctx.stroke();
ctx.lineWidth = 1;
// Draw change marker.
ctx.beginPath();
ctx.arc(xStart, y, 3 /*r*/, 0 /*start angle*/, 2 * Math.PI /*end angle*/);
ctx.fill();
ctx.stroke();
// Draw the tooltip.
ctx.fillStyle = 'rgba(255, 255, 255, 0.8)';
ctx.fillRect(this.mouseXpos + 5, MARGIN_TOP, width + 16, RECT_HEIGHT);
ctx.fillStyle = 'hsl(200, 50%, 40%)';
ctx.textAlign = 'left';
ctx.textBaseline = 'middle';
ctx.fillText(text, this.mouseXpos + 8, MARGIN_TOP + RECT_HEIGHT / 2);
}
// Write the Y scale on the top left corner.
ctx.fillStyle = 'rgba(255, 255, 255, 0.6)';
ctx.fillRect(0, 0, 42, 16);
ctx.fillStyle = '#666';
ctx.textAlign = 'left';
ctx.textBaseline = 'alphabetic';
ctx.fillText(`${yLabel}`, 5, 14);
// If the cached trace slices don't fully cover the visible time range,
// show a gray rectangle with a "Loading..." label.
checkerboardExcept(
ctx,
this.getHeight(),
timeScale.timeToPx(visibleWindowTime.start),
timeScale.timeToPx(visibleWindowTime.end),
timeScale.timeToPx(data.start),
timeScale.timeToPx(data.end));
}
onMouseMove({x}: {x: number, y: number}) {
const data = this.data();
if (data === undefined) return;
this.mouseXpos = x;
const {timeScale} = globals.frontendLocalState;
const time = timeScale.pxToTime(x);
const [left, right] = searchSegment(data.timestamps, time);
this.hoveredTs = left === -1 ? undefined : data.timestamps[left];
this.hoveredTsEnd = right === -1 ? undefined : data.timestamps[right];
this.hoveredValue = left === -1 ? undefined : data.values[left];
}
onMouseOut() {
this.hoveredValue = undefined;
this.hoveredTs = undefined;
}
onMouseClick({x}: {x: number}) {
const data = this.data();
if (data === undefined) return false;
const {timeScale} = globals.frontendLocalState;
const time = timeScale.pxToTime(x);
const [left, right] = searchSegment(data.timestamps, time);
if (left === -1) {
return false;
} else {
const counterId = data.ids[left];
globals.makeSelection(Actions.selectCounter({
leftTs: toNs(data.timestamps[left]),
rightTs: right !== -1 ? toNs(data.timestamps[right]) : -1,
id: counterId,
trackId: this.trackState.id
}));
return true;
}
}
}
trackRegistry.register(CounterTrack);