blob: ad06611c3d1fbd6fa93115d24cd4df789c18c731 [file] [log] [blame]
// Copyright 2017 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
Timeline.TimelineHistoryManager = class {
constructor() {
/** @type {!Array<!Timeline.PerformanceModel>} */
this._recordings = [];
this._action = /** @type {!UI.Action} */ (UI.actionRegistry.action('timeline.show-history'));
/** @type {!Map<string, number>} */
this._nextNumberByDomain = new Map();
this._button = new Timeline.TimelineHistoryManager.ToolbarButton(this._action);
UI.ARIAUtils.markAsMenuButton(this._button.element);
this.clear();
this._allOverviews = [
{constructor: Timeline.TimelineEventOverviewResponsiveness, height: 3},
{constructor: Timeline.TimelineEventOverviewFrames, height: 16},
{constructor: Timeline.TimelineEventOverviewCPUActivity, height: 20},
{constructor: Timeline.TimelineEventOverviewNetwork, height: 8}
];
this._totalHeight = this._allOverviews.reduce((acc, entry) => acc + entry.height, 0);
this._enabled = true;
/** @type {?Timeline.PerformanceModel} */
this._lastActiveModel = null;
}
/**
* @param {!Timeline.PerformanceModel} performanceModel
*/
addRecording(performanceModel) {
this._lastActiveModel = performanceModel;
this._recordings.unshift(performanceModel);
this._buildPreview(performanceModel);
const modelTitle = this._title(performanceModel);
this._button.setText(modelTitle);
const buttonTitle = this._action.title();
UI.ARIAUtils.setAccessibleName(this._button.element, ls`Current Session: ${modelTitle}. ${buttonTitle}`);
this._updateState();
if (this._recordings.length <= Timeline.TimelineHistoryManager._maxRecordings) {
return;
}
const lruModel = this._recordings.reduce((a, b) => lastUsedTime(a) < lastUsedTime(b) ? a : b);
this._recordings.splice(this._recordings.indexOf(lruModel), 1);
lruModel.dispose();
/**
* @param {!Timeline.PerformanceModel} model
* @return {number}
*/
function lastUsedTime(model) {
return Timeline.TimelineHistoryManager._dataForModel(model).lastUsed;
}
}
/**
* @param {boolean} enabled
*/
setEnabled(enabled) {
this._enabled = enabled;
this._updateState();
}
button() {
return this._button;
}
clear() {
this._recordings.forEach(model => model.dispose());
this._recordings = [];
this._lastActiveModel = null;
this._updateState();
this._button.setText(Common.UIString('(no recordings)'));
this._nextNumberByDomain.clear();
}
/**
* @return {!Promise<?Timeline.PerformanceModel>}
*/
async showHistoryDropDown() {
if (this._recordings.length < 2 || !this._enabled) {
return null;
}
// DropDown.show() function finishes when the dropdown menu is closed via selection or losing focus
const model = await Timeline.TimelineHistoryManager.DropDown.show(
this._recordings, /** @type {!Timeline.PerformanceModel} */ (this._lastActiveModel), this._button.element);
if (!model) {
return null;
}
const index = this._recordings.indexOf(model);
if (index < 0) {
console.assert(false, `selected recording not found`);
return null;
}
this._setCurrentModel(model);
return model;
}
cancelIfShowing() {
Timeline.TimelineHistoryManager.DropDown.cancelIfShowing();
}
/**
* @param {number} direction
* @return {?Timeline.PerformanceModel}
*/
navigate(direction) {
if (!this._enabled || !this._lastActiveModel) {
return null;
}
const index = this._recordings.indexOf(this._lastActiveModel);
if (index < 0) {
return null;
}
const newIndex = Number.constrain(index + direction, 0, this._recordings.length - 1);
const model = this._recordings[newIndex];
this._setCurrentModel(model);
return model;
}
/**
* @param {!Timeline.PerformanceModel} model
*/
_setCurrentModel(model) {
Timeline.TimelineHistoryManager._dataForModel(model).lastUsed = Date.now();
this._lastActiveModel = model;
const modelTitle = this._title(model);
const buttonTitle = this._action.title();
this._button.setText(modelTitle);
UI.ARIAUtils.setAccessibleName(this._button.element, ls`Current Session: ${modelTitle}. ${buttonTitle}`);
}
_updateState() {
this._action.setEnabled(this._recordings.length > 1 && this._enabled);
}
/**
* @param {!Timeline.PerformanceModel} performanceModel
* @return {!Element}
*/
static _previewElement(performanceModel) {
const data = Timeline.TimelineHistoryManager._dataForModel(performanceModel);
const startedAt = performanceModel.recordStartTime();
data.time.textContent =
startedAt ? Common.UIString('(%s ago)', Timeline.TimelineHistoryManager._coarseAge(startedAt)) : '';
return data.preview;
}
/**
* @param {number} time
* @return {string}
*/
static _coarseAge(time) {
const seconds = Math.round((Date.now() - time) / 1000);
if (seconds < 50) {
return Common.UIString('moments');
}
const minutes = Math.round(seconds / 60);
if (minutes < 50) {
return Common.UIString('%s m', minutes);
}
const hours = Math.round(minutes / 60);
return Common.UIString('%s h', hours);
}
/**
* @param {!Timeline.PerformanceModel} performanceModel
* @return {string}
*/
_title(performanceModel) {
return Timeline.TimelineHistoryManager._dataForModel(performanceModel).title;
}
/**
* @param {!Timeline.PerformanceModel} performanceModel
*/
_buildPreview(performanceModel) {
const parsedURL = Common.ParsedURL.fromString(performanceModel.timelineModel().pageURL());
const domain = parsedURL ? parsedURL.host : '';
const sequenceNumber = this._nextNumberByDomain.get(domain) || 1;
const title = Common.UIString('%s #%d', domain, sequenceNumber);
this._nextNumberByDomain.set(domain, sequenceNumber + 1);
const timeElement = createElement('span');
const preview = createElementWithClass('div', 'preview-item vbox');
const data = {preview: preview, title: title, time: timeElement, lastUsed: Date.now()};
performanceModel[Timeline.TimelineHistoryManager._previewDataSymbol] = data;
preview.appendChild(this._buildTextDetails(performanceModel, title, timeElement));
const screenshotAndOverview = preview.createChild('div', 'hbox');
screenshotAndOverview.appendChild(this._buildScreenshotThumbnail(performanceModel));
screenshotAndOverview.appendChild(this._buildOverview(performanceModel));
return data.preview;
}
/**
* @param {!Timeline.PerformanceModel} performanceModel
* @param {string} title
* @param {!Element} timeElement
* @return {!Element}
*/
_buildTextDetails(performanceModel, title, timeElement) {
const container = createElementWithClass('div', 'text-details hbox');
const nameSpan = container.createChild('span', 'name');
nameSpan.textContent = title;
UI.ARIAUtils.setAccessibleName(nameSpan, title);
const tracingModel = performanceModel.tracingModel();
const duration = Number.millisToString(tracingModel.maximumRecordTime() - tracingModel.minimumRecordTime(), false);
const timeContainer = container.createChild('span', 'time');
timeContainer.appendChild(createTextNode(duration));
timeContainer.appendChild(timeElement);
return container;
}
/**
* @param {!Timeline.PerformanceModel} performanceModel
* @return {!Element}
*/
_buildScreenshotThumbnail(performanceModel) {
const container = createElementWithClass('div', 'screenshot-thumb');
const thumbnailAspectRatio = 3 / 2;
container.style.width = this._totalHeight * thumbnailAspectRatio + 'px';
container.style.height = this._totalHeight + 'px';
const filmStripModel = performanceModel.filmStripModel();
const lastFrame = filmStripModel.frames().peekLast();
if (!lastFrame) {
return container;
}
lastFrame.imageDataPromise()
.then(data => UI.loadImageFromData(data))
.then(image => image && container.appendChild(image));
return container;
}
/**
* @param {!Timeline.PerformanceModel} performanceModel
* @return {!Element}
*/
_buildOverview(performanceModel) {
const container = createElement('div');
container.style.width = Timeline.TimelineHistoryManager._previewWidth + 'px';
container.style.height = this._totalHeight + 'px';
const canvas = container.createChild('canvas');
canvas.width = window.devicePixelRatio * Timeline.TimelineHistoryManager._previewWidth;
canvas.height = window.devicePixelRatio * this._totalHeight;
const ctx = canvas.getContext('2d');
let yOffset = 0;
for (const overview of this._allOverviews) {
const timelineOverview = new overview.constructor();
timelineOverview.setCanvasSize(Timeline.TimelineHistoryManager._previewWidth, overview.height);
timelineOverview.setModel(performanceModel);
timelineOverview.update();
const sourceContext = timelineOverview.context();
const imageData = sourceContext.getImageData(0, 0, sourceContext.canvas.width, sourceContext.canvas.height);
ctx.putImageData(imageData, 0, yOffset);
yOffset += overview.height * window.devicePixelRatio;
}
return container;
}
/**
* @param {!Timeline.PerformanceModel} model
* @return {?Timeline.TimelineHistoryManager.PreviewData}
*/
static _dataForModel(model) {
return model[Timeline.TimelineHistoryManager._previewDataSymbol] || null;
}
};
/** @typedef {!{preview: !Element, time: !Element, lastUsed: number, title: string}} */
Timeline.TimelineHistoryManager.PreviewData;
Timeline.TimelineHistoryManager._maxRecordings = 5;
Timeline.TimelineHistoryManager._previewWidth = 450;
Timeline.TimelineHistoryManager._previewDataSymbol = Symbol('previewData');
/**
* @implements {UI.ListDelegate<!Timeline.PerformanceModel>}
*/
Timeline.TimelineHistoryManager.DropDown = class {
/**
* @param {!Array<!Timeline.PerformanceModel>} models
*/
constructor(models) {
this._glassPane = new UI.GlassPane();
this._glassPane.setSizeBehavior(UI.GlassPane.SizeBehavior.MeasureContent);
this._glassPane.setOutsideClickCallback(() => this._close(null));
this._glassPane.setPointerEventsBehavior(UI.GlassPane.PointerEventsBehavior.BlockedByGlassPane);
this._glassPane.setAnchorBehavior(UI.GlassPane.AnchorBehavior.PreferBottom);
this._glassPane.element.addEventListener('blur', () => this._close(null));
const shadowRoot =
UI.createShadowRootWithCoreStyles(this._glassPane.contentElement, 'timeline/timelineHistoryManager.css');
const contentElement = shadowRoot.createChild('div', 'drop-down');
const listModel = new UI.ListModel();
this._listControl = new UI.ListControl(listModel, this, UI.ListMode.NonViewport);
this._listControl.element.addEventListener('mousemove', this._onMouseMove.bind(this), false);
listModel.replaceAll(models);
UI.ARIAUtils.markAsMenu(this._listControl.element);
UI.ARIAUtils.setAccessibleName(this._listControl.element, ls`Select Timeline Session`);
contentElement.appendChild(this._listControl.element);
contentElement.addEventListener('keydown', this._onKeyDown.bind(this), false);
contentElement.addEventListener('click', this._onClick.bind(this), false);
this._focusRestorer = new UI.ElementFocusRestorer(this._listControl.element);
/** @type {?function(?Timeline.PerformanceModel)} */
this._selectionDone = null;
}
/**
* @param {!Array<!Timeline.PerformanceModel>} models
* @param {!Timeline.PerformanceModel} currentModel
* @param {!Element} anchor
* @return {!Promise<?Timeline.PerformanceModel>}
*/
static show(models, currentModel, anchor) {
if (Timeline.TimelineHistoryManager.DropDown._instance) {
return Promise.resolve(/** @type {?Timeline.PerformanceModel} */ (null));
}
const instance = new Timeline.TimelineHistoryManager.DropDown(models);
return instance._show(anchor, currentModel);
}
static cancelIfShowing() {
if (!Timeline.TimelineHistoryManager.DropDown._instance) {
return;
}
Timeline.TimelineHistoryManager.DropDown._instance._close(null);
}
/**
* @param {!Element} anchor
* @param {!Timeline.PerformanceModel} currentModel
* @return {!Promise<?Timeline.PerformanceModel>}
*/
_show(anchor, currentModel) {
Timeline.TimelineHistoryManager.DropDown._instance = this;
this._glassPane.setContentAnchorBox(anchor.boxInWindow());
this._glassPane.show(/** @type {!Document} */ (this._glassPane.contentElement.ownerDocument));
this._listControl.element.focus();
this._listControl.selectItem(currentModel);
return new Promise(fulfill => this._selectionDone = fulfill);
}
/**
* @param {!Event} event
*/
_onMouseMove(event) {
const node = event.target.enclosingNodeOrSelfWithClass('preview-item');
const listItem = node && this._listControl.itemForNode(node);
if (!listItem) {
return;
}
this._listControl.selectItem(listItem);
}
/**
* @param {!Event} event
*/
_onClick(event) {
if (!event.target.enclosingNodeOrSelfWithClass('preview-item')) {
return;
}
this._close(this._listControl.selectedItem());
}
/**
* @param {!Event} event
*/
_onKeyDown(event) {
switch (event.key) {
case 'Tab':
case 'Escape':
this._close(null);
break;
case 'Enter':
this._close(this._listControl.selectedItem());
break;
default:
return;
}
event.consume(true);
}
/**
* @param {?Timeline.PerformanceModel} model
*/
_close(model) {
this._selectionDone(model);
this._focusRestorer.restore();
this._glassPane.hide();
Timeline.TimelineHistoryManager.DropDown._instance = null;
}
/**
* @override
* @param {!Timeline.PerformanceModel} item
* @return {!Element}
*/
createElementForItem(item) {
const element = Timeline.TimelineHistoryManager._previewElement(item);
UI.ARIAUtils.markAsMenuItem(element);
element.classList.remove('selected');
return element;
}
/**
* @override
* @param {!Timeline.PerformanceModel} item
* @return {number}
*/
heightForItem(item) {
console.assert(false, 'Should not be called');
return 0;
}
/**
* @override
* @param {!Timeline.PerformanceModel} item
* @return {boolean}
*/
isItemSelectable(item) {
return true;
}
/**
* @override
* @param {?Timeline.PerformanceModel} from
* @param {?Timeline.PerformanceModel} to
* @param {?Element} fromElement
* @param {?Element} toElement
*/
selectedItemChanged(from, to, fromElement, toElement) {
if (fromElement) {
fromElement.classList.remove('selected');
}
if (toElement) {
toElement.classList.add('selected');
}
}
/**
* @override
* @param {?Element} fromElement
* @param {?Element} toElement
* @return {boolean}
*/
updateSelectedItemARIA(fromElement, toElement) {
return false;
}
};
/**
* @type {?Timeline.TimelineHistoryManager.DropDown}
*/
Timeline.TimelineHistoryManager.DropDown._instance = null;
Timeline.TimelineHistoryManager.ToolbarButton = class extends UI.ToolbarItem {
/**
* @param {!UI.Action} action
*/
constructor(action) {
super(createElementWithClass('button', 'history-dropdown-button'));
UI.appendStyle(this.element, 'timeline/historyToolbarButton.css');
this._contentElement = this.element.createChild('span', 'content');
const dropdownArrowIcon = UI.Icon.create('smallicon-triangle-down');
this.element.appendChild(dropdownArrowIcon);
this.element.addEventListener('click', () => void action.execute(), false);
this.setEnabled(action.enabled());
action.addEventListener(UI.Action.Events.Enabled, event => this.setEnabled(/** @type {boolean} */ (event.data)));
this.setTitle(action.title());
}
/**
* @param {string} text
*/
setText(text) {
this._contentElement.textContent = text;
}
};