mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
7a653b042a
--HG-- rename : browser/devtools/shared/test/browser_graphs-07.js => browser/devtools/shared/test/browser_graphs-07a.js
209 lines
7.0 KiB
JavaScript
209 lines
7.0 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
"use strict";
|
|
|
|
/**
|
|
* This file contains the "overview" graph, which is a minimap of all the
|
|
* timeline data. Regions inside it may be selected, determining which markers
|
|
* are visible in the "waterfall".
|
|
*/
|
|
|
|
const {Cc, Ci, Cu, Cr} = require("chrome");
|
|
|
|
Cu.import("resource:///modules/devtools/Graphs.jsm");
|
|
Cu.import("resource:///modules/devtools/ViewHelpers.jsm");
|
|
|
|
loader.lazyRequireGetter(this, "L10N",
|
|
"devtools/timeline/global", true);
|
|
loader.lazyRequireGetter(this, "TIMELINE_BLUEPRINT",
|
|
"devtools/timeline/global", true);
|
|
|
|
const HTML_NS = "http://www.w3.org/1999/xhtml";
|
|
|
|
const OVERVIEW_HEADER_HEIGHT = 20; // px
|
|
const OVERVIEW_BODY_HEIGHT = 50; // px
|
|
|
|
const OVERVIEW_BACKGROUND_COLOR = "#fff";
|
|
const OVERVIEW_CLIPHEAD_LINE_COLOR = "#666";
|
|
const OVERVIEW_SELECTION_LINE_COLOR = "#555";
|
|
const OVERVIEW_SELECTION_BACKGROUND_COLOR = "rgba(76,158,217,0.25)";
|
|
const OVERVIEW_SELECTION_STRIPES_COLOR = "rgba(255,255,255,0.1)";
|
|
|
|
const OVERVIEW_HEADER_TICKS_MULTIPLE = 100; // ms
|
|
const OVERVIEW_HEADER_TICKS_SPACING_MIN = 75; // px
|
|
const OVERVIEW_HEADER_SAFE_BOUNDS = 50; // px
|
|
const OVERVIEW_HEADER_BACKGROUND = "#ebeced";
|
|
const OVERVIEW_HEADER_TEXT_COLOR = "#18191a";
|
|
const OVERVIEW_HEADER_TEXT_FONT_SIZE = 9; // px
|
|
const OVERVIEW_HEADER_TEXT_FONT_FAMILY = "sans-serif";
|
|
const OVERVIEW_HEADER_TEXT_PADDING = 6; // px
|
|
const OVERVIEW_TIMELINE_STROKES = "#aaa";
|
|
const OVERVIEW_MARKERS_COLOR_STOPS = [0, 0.1, 0.75, 1];
|
|
const OVERVIEW_MARKER_DURATION_MIN = 4; // ms
|
|
const OVERVIEW_GROUP_VERTICAL_PADDING = 6; // px
|
|
const OVERVIEW_GROUP_ALTERNATING_BACKGROUND = "rgba(0,0,0,0.05)";
|
|
|
|
/**
|
|
* An overview for the timeline data.
|
|
*
|
|
* @param nsIDOMNode parent
|
|
* The parent node holding the overview.
|
|
*/
|
|
function Overview(parent, ...args) {
|
|
AbstractCanvasGraph.apply(this, [parent, "timeline-overview", ...args]);
|
|
this.once("ready", () => {
|
|
this.setBlueprint(TIMELINE_BLUEPRINT);
|
|
|
|
var preview = [];
|
|
preview.startTime = 0;
|
|
preview.endTime = 1000;
|
|
this.setData(preview);
|
|
});
|
|
}
|
|
|
|
Overview.prototype = Heritage.extend(AbstractCanvasGraph.prototype, {
|
|
fixedHeight: OVERVIEW_HEADER_HEIGHT + OVERVIEW_BODY_HEIGHT,
|
|
clipheadLineColor: OVERVIEW_CLIPHEAD_LINE_COLOR,
|
|
selectionLineColor: OVERVIEW_SELECTION_LINE_COLOR,
|
|
selectionBackgroundColor: OVERVIEW_SELECTION_BACKGROUND_COLOR,
|
|
selectionStripesColor: OVERVIEW_SELECTION_STRIPES_COLOR,
|
|
|
|
/**
|
|
* List of names and colors used to paint this overview.
|
|
* @see TIMELINE_BLUEPRINT in timeline/widgets/global.js
|
|
*/
|
|
setBlueprint: function(blueprint) {
|
|
this._paintBatches = new Map();
|
|
this._lastGroup = 0;
|
|
|
|
for (let type in blueprint) {
|
|
this._paintBatches.set(type, { style: blueprint[type], batch: [] });
|
|
this._lastGroup = Math.max(this._lastGroup, blueprint[type].group);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Renders the graph's data source.
|
|
* @see AbstractCanvasGraph.prototype.buildGraphImage
|
|
*/
|
|
buildGraphImage: function() {
|
|
let { canvas, ctx } = this._getNamedCanvas("overview-data");
|
|
let canvasWidth = this._width;
|
|
let canvasHeight = this._height;
|
|
let safeBounds = OVERVIEW_HEADER_SAFE_BOUNDS * this._pixelRatio;
|
|
let availableWidth = canvasWidth - safeBounds;
|
|
|
|
// Group markers into separate paint batches. This is necessary to
|
|
// draw all markers sharing the same style at once.
|
|
|
|
for (let marker of this._data) {
|
|
this._paintBatches.get(marker.name).batch.push(marker);
|
|
}
|
|
|
|
// Calculate each group's height, and the time-based scaling.
|
|
|
|
let totalGroups = this._lastGroup + 1;
|
|
let headerHeight = OVERVIEW_HEADER_HEIGHT * this._pixelRatio;
|
|
let groupHeight = OVERVIEW_BODY_HEIGHT * this._pixelRatio / totalGroups;
|
|
let groupPadding = OVERVIEW_GROUP_VERTICAL_PADDING * this._pixelRatio;
|
|
|
|
let totalTime = (this._data.endTime - this._data.startTime) || 0;
|
|
let dataScale = this.dataScaleX = availableWidth / totalTime;
|
|
|
|
// Draw the header and overview background.
|
|
|
|
ctx.fillStyle = OVERVIEW_HEADER_BACKGROUND;
|
|
ctx.fillRect(0, 0, canvasWidth, headerHeight);
|
|
|
|
ctx.fillStyle = OVERVIEW_BACKGROUND_COLOR;
|
|
ctx.fillRect(0, headerHeight, canvasWidth, canvasHeight);
|
|
|
|
// Draw the alternating odd/even group backgrounds.
|
|
|
|
ctx.fillStyle = OVERVIEW_GROUP_ALTERNATING_BACKGROUND;
|
|
ctx.beginPath();
|
|
|
|
for (let i = 1; i < totalGroups; i += 2) {
|
|
let top = headerHeight + i * groupHeight;
|
|
ctx.rect(0, top, canvasWidth, groupHeight);
|
|
}
|
|
|
|
ctx.fill();
|
|
|
|
// Draw the timeline header ticks.
|
|
|
|
ctx.textBaseline = "middle";
|
|
let fontSize = OVERVIEW_HEADER_TEXT_FONT_SIZE * this._pixelRatio;
|
|
let fontFamily = OVERVIEW_HEADER_TEXT_FONT_FAMILY;
|
|
ctx.font = fontSize + "px " + fontFamily;
|
|
ctx.fillStyle = OVERVIEW_HEADER_TEXT_COLOR;
|
|
ctx.strokeStyle = OVERVIEW_TIMELINE_STROKES;
|
|
ctx.beginPath();
|
|
|
|
let tickInterval = this._findOptimalTickInterval(dataScale);
|
|
let headerTextPadding = OVERVIEW_HEADER_TEXT_PADDING * this._pixelRatio;
|
|
|
|
for (let x = 0; x < availableWidth; x += tickInterval) {
|
|
let left = x + headerTextPadding;
|
|
let time = Math.round(x / dataScale);
|
|
let label = L10N.getFormatStr("timeline.tick", time);
|
|
ctx.fillText(label, left, headerHeight / 2 + 1);
|
|
ctx.moveTo(x, 0);
|
|
ctx.lineTo(x, canvasHeight);
|
|
}
|
|
|
|
ctx.stroke();
|
|
|
|
// Draw the timeline markers.
|
|
|
|
for (let [, { style, batch }] of this._paintBatches) {
|
|
let top = headerHeight + style.group * groupHeight + groupPadding / 2;
|
|
let height = groupHeight - groupPadding;
|
|
|
|
let gradient = ctx.createLinearGradient(0, top, 0, top + height);
|
|
gradient.addColorStop(OVERVIEW_MARKERS_COLOR_STOPS[0], style.stroke);
|
|
gradient.addColorStop(OVERVIEW_MARKERS_COLOR_STOPS[1], style.fill);
|
|
gradient.addColorStop(OVERVIEW_MARKERS_COLOR_STOPS[2], style.fill);
|
|
gradient.addColorStop(OVERVIEW_MARKERS_COLOR_STOPS[3], style.stroke);
|
|
ctx.fillStyle = gradient;
|
|
ctx.beginPath();
|
|
|
|
for (let { start, end } of batch) {
|
|
let left = start * dataScale;
|
|
let duration = Math.max(end - start, OVERVIEW_MARKER_DURATION_MIN);
|
|
let width = Math.max(duration * dataScale, this._pixelRatio);
|
|
ctx.rect(left, top, width, height);
|
|
}
|
|
|
|
ctx.fill();
|
|
|
|
// Since all the markers in this batch (thus sharing the same style) have
|
|
// been drawn, empty it. The next time new markers will be available,
|
|
// they will be sorted and drawn again.
|
|
batch.length = 0;
|
|
}
|
|
|
|
return canvas;
|
|
},
|
|
|
|
/**
|
|
* Finds the optimal tick interval between time markers in this overview.
|
|
*/
|
|
_findOptimalTickInterval: function(dataScale) {
|
|
let timingStep = OVERVIEW_HEADER_TICKS_MULTIPLE;
|
|
let spacingMin = OVERVIEW_HEADER_TICKS_SPACING_MIN * this._pixelRatio;
|
|
|
|
while (true) {
|
|
let scaledStep = dataScale * timingStep;
|
|
if (scaledStep < spacingMin) {
|
|
timingStep <<= 1;
|
|
continue;
|
|
}
|
|
return scaledStep;
|
|
}
|
|
}
|
|
});
|
|
|
|
exports.Overview = Overview;
|