blob: 7c76711877ae8aff3ccf71488f245a9e1014a75f [file] [log] [blame]
/*
* Copyright (C) 2013, 2015 Apple Inc. All rights reserved.
* Copyright (C) 2015 University of Washington.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
WI.TimelineRecordingContentView = class TimelineRecordingContentView extends WI.ContentView
{
constructor(recording)
{
super(recording);
this._recording = recording;
this.element.classList.add("timeline-recording");
this._timelineOverview = new WI.TimelineOverview(this._recording);
this._timelineOverview.addEventListener(WI.TimelineOverview.Event.TimeRangeSelectionChanged, this._timeRangeSelectionChanged, this);
this._timelineOverview.addEventListener(WI.TimelineOverview.Event.RecordSelected, this._recordSelected, this);
this._timelineOverview.addEventListener(WI.TimelineOverview.Event.TimelineSelected, this._timelineSelected, this);
this._timelineOverview.addEventListener(WI.TimelineOverview.Event.EditingInstrumentsDidChange, this._editingInstrumentsDidChange, this);
this.addSubview(this._timelineOverview);
const disableBackForward = true;
const disableFindBanner = true;
this._timelineContentBrowser = new WI.ContentBrowser(null, this, disableBackForward, disableFindBanner);
this._timelineContentBrowser.addEventListener(WI.ContentBrowser.Event.CurrentContentViewDidChange, this._currentContentViewDidChange, this);
this._entireRecordingPathComponent = this._createTimelineRangePathComponent(WI.UIString("Entire Recording"));
this._timelineSelectionPathComponent = this._createTimelineRangePathComponent();
this._timelineSelectionPathComponent.previousSibling = this._entireRecordingPathComponent;
this._selectedTimeRangePathComponent = this._entireRecordingPathComponent;
this._filterBarNavigationItem = new WI.FilterBarNavigationItem;
this._filterBarNavigationItem.filterBar.addEventListener(WI.FilterBar.Event.FilterDidChange, this._filterDidChange, this);
this._timelineContentBrowser.navigationBar.addNavigationItem(this._filterBarNavigationItem);
this.addSubview(this._timelineContentBrowser);
if (WI.sharedApp.debuggableType === WI.DebuggableType.Web) {
this._autoStopCheckboxNavigationItem = new WI.CheckboxNavigationItem("auto-stop-recording", WI.UIString("Stop recording once page loads"), WI.settings.timelinesAutoStop.value);
this._autoStopCheckboxNavigationItem.visibilityPriority = WI.NavigationItem.VisibilityPriority.Low;
this._autoStopCheckboxNavigationItem.addEventListener(WI.CheckboxNavigationItem.Event.CheckedDidChange, this._handleAutoStopCheckboxCheckedDidChange, this);
WI.settings.timelinesAutoStop.addEventListener(WI.Setting.Event.Changed, this._handleTimelinesAutoStopSettingChanged, this);
}
this._exportButtonNavigationItem = new WI.ButtonNavigationItem("export", WI.UIString("Export"), "Images/Export.svg", 15, 15);
this._exportButtonNavigationItem.toolTip = WI.UIString("Export (%s)").format(WI.saveKeyboardShortcut.displayName);
this._exportButtonNavigationItem.buttonStyle = WI.ButtonNavigationItem.Style.ImageAndText;
this._exportButtonNavigationItem.visibilityPriority = WI.NavigationItem.VisibilityPriority.Low;
this._exportButtonNavigationItem.addEventListener(WI.ButtonNavigationItem.Event.Clicked, this._exportButtonNavigationItemClicked, this);
this._exportButtonNavigationItem.enabled = false;
this._importButtonNavigationItem = new WI.ButtonNavigationItem("import", WI.UIString("Import"), "Images/Import.svg", 15, 15);
this._importButtonNavigationItem.toolTip = WI.UIString("Import");
this._importButtonNavigationItem.buttonStyle = WI.ButtonNavigationItem.Style.ImageAndText;
this._importButtonNavigationItem.visibilityPriority = WI.NavigationItem.VisibilityPriority.Low;
this._importButtonNavigationItem.addEventListener(WI.ButtonNavigationItem.Event.Clicked, this._importButtonNavigationItemClicked, this);
this._clearTimelineNavigationItem = new WI.ButtonNavigationItem("clear-timeline", WI.UIString("Clear Timeline (%s)").format(WI.clearKeyboardShortcut.displayName), "Images/NavigationItemTrash.svg", 15, 15);
this._clearTimelineNavigationItem.visibilityPriority = WI.NavigationItem.VisibilityPriority.Low;
this._clearTimelineNavigationItem.addEventListener(WI.ButtonNavigationItem.Event.Clicked, this._clearTimeline, this);
this._overviewTimelineView = new WI.OverviewTimelineView(recording);
this._overviewTimelineView.secondsPerPixel = this._timelineOverview.secondsPerPixel;
this._progressView = new WI.TimelineRecordingProgressView;
this._timelineContentBrowser.addSubview(this._progressView);
this._timelineViewMap = new Map;
this._pathComponentMap = new Map;
this._updating = false;
this._currentTime = NaN;
this._lastUpdateTimestamp = NaN;
this._startTimeNeedsReset = true;
this._renderingFrameTimeline = null;
this._recording.addEventListener(WI.TimelineRecording.Event.InstrumentAdded, this._instrumentAdded, this);
this._recording.addEventListener(WI.TimelineRecording.Event.InstrumentRemoved, this._instrumentRemoved, this);
this._recording.addEventListener(WI.TimelineRecording.Event.Reset, this._recordingReset, this);
this._recording.addEventListener(WI.TimelineRecording.Event.Unloaded, this._recordingUnloaded, this);
WI.timelineManager.addEventListener(WI.TimelineManager.Event.CapturingStateChanged, this._handleTimelineCapturingStateChanged, this);
WI.debuggerManager.addEventListener(WI.DebuggerManager.Event.Paused, this._debuggerPaused, this);
WI.debuggerManager.addEventListener(WI.DebuggerManager.Event.Resumed, this._debuggerResumed, this);
WI.ContentView.addEventListener(WI.ContentView.Event.SelectionPathComponentsDidChange, this._contentViewSelectionPathComponentDidChange, this);
WI.ContentView.addEventListener(WI.ContentView.Event.SupplementalRepresentedObjectsDidChange, this._contentViewSupplementalRepresentedObjectsDidChange, this);
WI.TimelineView.addEventListener(WI.TimelineView.Event.RecordWasFiltered, this._handleTimelineViewRecordFiltered, this);
WI.TimelineView.addEventListener(WI.TimelineView.Event.RecordWasSelected, this._handleTimelineViewRecordSelected, this);
WI.TimelineView.addEventListener(WI.TimelineView.Event.ScannerShow, this._handleTimelineViewScannerShow, this);
WI.TimelineView.addEventListener(WI.TimelineView.Event.ScannerHide, this._handleTimelineViewScannerHide, this);
WI.TimelineView.addEventListener(WI.TimelineView.Event.NeedsEntireSelectedRange, this._handleTimelineViewNeedsEntireSelectedRange, this);
WI.notifications.addEventListener(WI.Notification.VisibilityStateDidChange, this._inspectorVisibilityStateChanged, this);
for (let instrument of this._recording.instruments)
this._instrumentAdded(instrument);
this.showOverviewTimelineView();
if (this._recording.imported) {
let {startTime, endTime} = this._recording;
this._updateTimes(startTime, endTime, endTime);
}
}
// Public
showOverviewTimelineView()
{
this._timelineContentBrowser.showContentView(this._overviewTimelineView);
}
showTimelineViewForTimeline(timeline)
{
console.assert(timeline instanceof WI.Timeline, timeline);
console.assert(this._timelineViewMap.has(timeline), timeline);
if (!this._timelineViewMap.has(timeline))
return;
let contentView = this._timelineContentBrowser.showContentView(this._timelineViewMap.get(timeline));
// FIXME: `WI.HeapAllocationsTimelineView` relies on it's `_dataGrid` for determining what
// object is currently selected. If that `_dataGrid` hasn't yet called `layout()` when first
// shown, we will lose the selection.
if (!contentView.didInitialLayout)
contentView.updateLayout();
}
get supportsSplitContentBrowser()
{
// The layout of the overview and split content browser don't work well.
return false;
}
get selectionPathComponents()
{
if (!this._timelineContentBrowser.currentContentView)
return [];
let pathComponents = [];
let representedObject = this._timelineContentBrowser.currentContentView.representedObject;
if (representedObject instanceof WI.Timeline)
pathComponents.push(this._pathComponentMap.get(representedObject));
pathComponents.push(this._selectedTimeRangePathComponent);
return pathComponents;
}
get supplementalRepresentedObjects()
{
if (!this._timelineContentBrowser.currentContentView)
return [];
return this._timelineContentBrowser.currentContentView.supplementalRepresentedObjects;
}
get navigationItems()
{
let navigationItems = [];
if (this._autoStopCheckboxNavigationItem)
navigationItems.push(this._autoStopCheckboxNavigationItem);
navigationItems.push(new WI.DividerNavigationItem);
navigationItems.push(this._importButtonNavigationItem);
navigationItems.push(this._exportButtonNavigationItem);
navigationItems.push(new WI.DividerNavigationItem);
navigationItems.push(this._clearTimelineNavigationItem);
return navigationItems;
}
get handleCopyEvent()
{
let currentContentView = this._timelineContentBrowser.currentContentView;
return currentContentView && typeof currentContentView.handleCopyEvent === "function" ? currentContentView.handleCopyEvent.bind(currentContentView) : null;
}
get supportsSave()
{
return this._recording.canExport();
}
get saveData()
{
return {customSaveHandler: () => { this._exportTimelineRecording(); }};
}
get currentTimelineView()
{
return this._timelineContentBrowser.currentContentView;
}
shown()
{
super.shown();
this._timelineOverview.shown();
this._timelineContentBrowser.shown();
this._clearTimelineNavigationItem.enabled = !this._recording.readonly && !isNaN(this._recording.startTime);
this._exportButtonNavigationItem.enabled = this._recording.canExport();
this._currentContentViewDidChange();
if (!this._updating && WI.timelineManager.activeRecording === this._recording && WI.timelineManager.isCapturing())
this._startUpdatingCurrentTime(this._currentTime);
}
hidden()
{
super.hidden();
this._timelineOverview.hidden();
this._timelineContentBrowser.hidden();
if (this._updating)
this._stopUpdatingCurrentTime();
}
closed()
{
super.closed();
this._timelineContentBrowser.contentViewContainer.closeAllContentViews();
this._recording.removeEventListener(null, null, this);
WI.timelineManager.removeEventListener(null, null, this);
WI.debuggerManager.removeEventListener(null, null, this);
WI.ContentView.removeEventListener(null, null, this);
}
canGoBack()
{
return this._timelineContentBrowser.canGoBack();
}
canGoForward()
{
return this._timelineContentBrowser.canGoForward();
}
goBack()
{
this._timelineContentBrowser.goBack();
}
goForward()
{
this._timelineContentBrowser.goForward();
}
handleClearShortcut(event)
{
this._clearTimeline();
}
get canFocusFilterBar()
{
return !this._filterBarNavigationItem.hidden;
}
focusFilterBar()
{
this._filterBarNavigationItem.filterBar.focus();
}
// ContentBrowser delegate
contentBrowserTreeElementForRepresentedObject(contentBrowser, representedObject)
{
if (!(representedObject instanceof WI.Timeline) && !(representedObject instanceof WI.TimelineRecording))
return null;
let iconClassName;
let title;
if (representedObject instanceof WI.Timeline) {
iconClassName = WI.TimelineTabContentView.iconClassNameForTimelineType(representedObject.type);
title = WI.UIString("Details");
} else {
iconClassName = WI.TimelineTabContentView.StopwatchIconStyleClass;
title = WI.UIString("Overview");
}
const hasChildren = false;
return new WI.GeneralTreeElement(iconClassName, title, representedObject, hasChildren);
}
// Private
_currentContentViewDidChange(event)
{
let newViewMode;
let timelineView = this.currentTimelineView;
if (timelineView && timelineView.representedObject.type === WI.TimelineRecord.Type.RenderingFrame)
newViewMode = WI.TimelineOverview.ViewMode.RenderingFrames;
else
newViewMode = WI.TimelineOverview.ViewMode.Timelines;
this._timelineOverview.viewMode = newViewMode;
this._updateTimelineOverviewHeight();
this._updateProgressView();
this._updateFilterBar();
if (timelineView) {
this._updateTimelineViewTimes(timelineView);
this._filterDidChange();
let timeline = null;
if (timelineView.representedObject instanceof WI.Timeline)
timeline = timelineView.representedObject;
this._timelineOverview.selectedTimeline = timeline;
}
this.dispatchEventToListeners(WI.ContentView.Event.SelectionPathComponentsDidChange);
this.dispatchEventToListeners(WI.ContentView.Event.NavigationItemsDidChange);
}
_timelinePathComponentSelected(event)
{
let selectedTimeline = event.data.pathComponent.representedObject;
this.showTimelineViewForTimeline(selectedTimeline);
}
_timeRangePathComponentSelected(event)
{
let selectedPathComponent = event.data.pathComponent;
if (selectedPathComponent === this._selectedTimeRangePathComponent)
return;
let timelineRuler = this._timelineOverview.timelineRuler;
if (selectedPathComponent === this._entireRecordingPathComponent)
timelineRuler.selectEntireRange();
else {
let timelineRange = selectedPathComponent.representedObject;
timelineRuler.selectionStartTime = timelineRuler.zeroTime + timelineRange.startValue;
timelineRuler.selectionEndTime = timelineRuler.zeroTime + timelineRange.endValue;
}
}
_contentViewSelectionPathComponentDidChange(event)
{
if (!this.visible)
return;
if (event.target !== this._timelineContentBrowser.currentContentView)
return;
this._updateFilterBar();
this.dispatchEventToListeners(WI.ContentView.Event.SelectionPathComponentsDidChange);
if (this.currentTimelineView === this._overviewTimelineView)
return;
let record = null;
if (this.currentTimelineView.selectionPathComponents) {
let recordPathComponent = this.currentTimelineView.selectionPathComponents.find((element) => element.representedObject instanceof WI.TimelineRecord);
record = recordPathComponent ? recordPathComponent.representedObject : null;
}
this._timelineOverview.selectRecord(event.target.representedObject, record);
}
_contentViewSupplementalRepresentedObjectsDidChange(event)
{
if (event.target !== this._timelineContentBrowser.currentContentView)
return;
this.dispatchEventToListeners(WI.ContentView.Event.SupplementalRepresentedObjectsDidChange);
}
_inspectorVisibilityStateChanged()
{
if (WI.timelineManager.activeRecording !== this._recording)
return;
// Stop updating since the results won't be rendered anyway.
if (!WI.visible && this._updating) {
this._stopUpdatingCurrentTime();
return;
}
// Nothing else to do if the current time was not being updated.
if (!WI.visible)
return;
let {startTime, endTime} = this.representedObject;
if (!WI.timelineManager.isCapturing()) {
// Force the overview to render data from the entire recording.
// This is necessary if the recording was started when the inspector was not
// visible because the views were never updated with currentTime/endTime.
this._updateTimes(startTime, endTime, endTime);
return;
}
this._startUpdatingCurrentTime(endTime);
}
_update(timestamp)
{
// FIXME: <https://webkit.org/b/153634> Web Inspector: some background tabs think they are the foreground tab and do unnecessary work
if (!(WI.tabBrowser.selectedTabContentView instanceof WI.TimelineTabContentView))
return;
if (this._waitingToResetCurrentTime) {
requestAnimationFrame(this._updateCallback);
return;
}
var startTime = this._recording.startTime;
var currentTime = this._currentTime || startTime;
var endTime = this._recording.endTime;
var timespanSinceLastUpdate = (timestamp - this._lastUpdateTimestamp) / 1000 || 0;
currentTime += timespanSinceLastUpdate;
this._updateTimes(startTime, currentTime, endTime);
// Only stop updating if the current time is greater than the end time, or the end time is NaN.
// The recording end time will be NaN if no records were added.
if (!this._updating && (currentTime >= endTime || isNaN(endTime))) {
if (this.visible)
this._lastUpdateTimestamp = NaN;
return;
}
this._lastUpdateTimestamp = timestamp;
requestAnimationFrame(this._updateCallback);
}
_updateTimes(startTime, currentTime, endTime)
{
if (this._startTimeNeedsReset && !isNaN(startTime)) {
this._timelineOverview.startTime = startTime;
this._overviewTimelineView.zeroTime = startTime;
for (let timelineView of this._timelineViewMap.values())
timelineView.zeroTime = startTime;
this._startTimeNeedsReset = false;
}
this._timelineOverview.endTime = Math.max(endTime, currentTime);
this._currentTime = currentTime;
this._timelineOverview.currentTime = currentTime;
if (this.currentTimelineView)
this._updateTimelineViewTimes(this.currentTimelineView);
// Force a layout now since we are already in an animation frame and don't need to delay it until the next.
this._timelineOverview.updateLayoutIfNeeded();
if (this.currentTimelineView)
this.currentTimelineView.updateLayoutIfNeeded();
}
_startUpdatingCurrentTime(startTime)
{
console.assert(!this._updating);
if (this._updating)
return;
// Don't update the current time if the Inspector is not visible, as the requestAnimationFrames won't work.
if (!WI.visible)
return;
if (typeof startTime === "number")
this._currentTime = startTime;
else if (!isNaN(this._currentTime)) {
// This happens when you stop and later restart recording.
// COMPATIBILITY (iOS 9): Timeline.recordingStarted events did not include a timestamp.
// We likely need to jump into the future to a better current time which we can
// ascertained from a new incoming timeline record, so we wait for a Timeline to update.
console.assert(!this._waitingToResetCurrentTime);
this._waitingToResetCurrentTime = true;
this._recording.addEventListener(WI.TimelineRecording.Event.TimesUpdated, this._recordingTimesUpdated, this);
}
this._updating = true;
if (!this._updateCallback)
this._updateCallback = this._update.bind(this);
requestAnimationFrame(this._updateCallback);
}
_stopUpdatingCurrentTime()
{
console.assert(this._updating);
this._updating = false;
if (this._waitingToResetCurrentTime) {
// Did not get any event while waiting for the current time, but we should stop waiting.
this._recording.removeEventListener(WI.TimelineRecording.Event.TimesUpdated, this._recordingTimesUpdated, this);
this._waitingToResetCurrentTime = false;
}
}
_handleTimelineCapturingStateChanged(event)
{
let {startTime, endTime} = event.data;
this._updateProgressView();
switch (WI.timelineManager.capturingState) {
case WI.TimelineManager.CapturingState.Active:
if (!this._updating)
this._startUpdatingCurrentTime(startTime);
this._clearTimelineNavigationItem.enabled = !this._recording.readonly;
this._exportButtonNavigationItem.enabled = false;
break;
case WI.TimelineManager.CapturingState.Inactive:
if (this._updating)
this._stopUpdatingCurrentTime();
if (this.currentTimelineView)
this._updateTimelineViewTimes(this.currentTimelineView);
this._exportButtonNavigationItem.enabled = this._recording.canExport();
break;
}
}
_debuggerPaused(event)
{
if (this._updating)
this._stopUpdatingCurrentTime();
}
_debuggerResumed(event)
{
if (!this._updating)
this._startUpdatingCurrentTime();
}
_recordingTimesUpdated(event)
{
if (!this._waitingToResetCurrentTime)
return;
// COMPATIBILITY (iOS 9): Timeline.recordingStarted events did not include a new startTime.
// Make the current time be the start time of the last added record. This is the best way
// currently to jump to the right period of time after recording starts.
for (var timeline of this._recording.timelines.values()) {
var lastRecord = timeline.records.lastValue;
if (!lastRecord)
continue;
this._currentTime = Math.max(this._currentTime, lastRecord.startTime);
}
this._recording.removeEventListener(WI.TimelineRecording.Event.TimesUpdated, this._recordingTimesUpdated, this);
this._waitingToResetCurrentTime = false;
}
_handleAutoStopCheckboxCheckedDidChange(event)
{
WI.settings.timelinesAutoStop.value = this._autoStopCheckboxNavigationItem.checked;
}
_handleTimelinesAutoStopSettingChanged(event)
{
this._autoStopCheckboxNavigationItem.checked = WI.settings.timelinesAutoStop.value;
}
_exportTimelineRecording()
{
let json = {
version: WI.TimelineRecording.SerializationVersion,
recording: this._recording.exportData(),
overview: this._timelineOverview.exportData(),
};
if (!json.recording || !json.overview) {
InspectorFrontendHost.beep();
return;
}
let frameName = null;
let mainFrame = WI.networkManager.mainFrame;
if (mainFrame)
frameName = mainFrame.mainResource.urlComponents.host || mainFrame.mainResource.displayName;
let filename = frameName ? `${frameName}-recording` : this._recording.displayName;
WI.FileUtilities.save({
url: WI.FileUtilities.inspectorURLForFilename(filename + ".json"),
content: JSON.stringify(json),
forceSaveAs: true,
});
}
_exportButtonNavigationItemClicked(event)
{
this._exportTimelineRecording();
}
_importButtonNavigationItemClicked(event)
{
WI.FileUtilities.importJSON((result) => WI.timelineManager.processJSON(result));
}
_clearTimeline(event)
{
if (this._recording.readonly)
return;
if (WI.timelineManager.activeRecording === this._recording && WI.timelineManager.isCapturing())
WI.timelineManager.stopCapturing();
this._recording.reset();
}
_updateTimelineOverviewHeight()
{
if (this._timelineOverview.editingInstruments)
this._timelineOverview.element.style.height = "";
else {
const rulerHeight = 23;
let styleValue = (rulerHeight + this._timelineOverview.height) + "px";
this._timelineOverview.element.style.height = styleValue;
this._timelineContentBrowser.element.style.top = styleValue;
}
}
_instrumentAdded(instrumentOrEvent)
{
let instrument = instrumentOrEvent instanceof WI.Instrument ? instrumentOrEvent : instrumentOrEvent.data.instrument;
console.assert(instrument instanceof WI.Instrument, instrument);
let timeline = this._recording.timelineForInstrument(instrument);
console.assert(!this._timelineViewMap.has(timeline), timeline);
this._timelineViewMap.set(timeline, WI.ContentView.createFromRepresentedObject(timeline, {recording: this._recording}));
if (timeline.type === WI.TimelineRecord.Type.RenderingFrame)
this._renderingFrameTimeline = timeline;
let displayName = WI.TimelineTabContentView.displayNameForTimelineType(timeline.type);
let iconClassName = WI.TimelineTabContentView.iconClassNameForTimelineType(timeline.type);
let pathComponent = new WI.HierarchicalPathComponent(displayName, iconClassName, timeline);
pathComponent.addEventListener(WI.HierarchicalPathComponent.Event.SiblingWasSelected, this._timelinePathComponentSelected, this);
this._pathComponentMap.set(timeline, pathComponent);
this._timelineCountChanged();
}
_instrumentRemoved(event)
{
let instrument = event.data.instrument;
console.assert(instrument instanceof WI.Instrument);
let timeline = this._recording.timelineForInstrument(instrument);
console.assert(this._timelineViewMap.has(timeline), timeline);
let timelineView = this._timelineViewMap.take(timeline);
if (this.currentTimelineView === timelineView)
this.showOverviewTimelineView();
if (timeline.type === WI.TimelineRecord.Type.RenderingFrame)
this._renderingFrameTimeline = null;
this._pathComponentMap.delete(timeline);
this._timelineCountChanged();
}
_timelineCountChanged()
{
var previousPathComponent = null;
for (var pathComponent of this._pathComponentMap.values()) {
if (previousPathComponent) {
previousPathComponent.nextSibling = pathComponent;
pathComponent.previousSibling = previousPathComponent;
}
previousPathComponent = pathComponent;
}
this._updateTimelineOverviewHeight();
}
_recordingReset(event)
{
for (let timelineView of this._timelineViewMap.values())
timelineView.reset();
this._currentTime = NaN;
if (!this._updating) {
// Force the time ruler and views to reset to 0.
this._startTimeNeedsReset = true;
this._updateTimes(0, 0, 0);
}
this._lastUpdateTimestamp = NaN;
this._startTimeNeedsReset = true;
this._recording.removeEventListener(WI.TimelineRecording.Event.TimesUpdated, this._recordingTimesUpdated, this);
this._waitingToResetCurrentTime = false;
this._timelineOverview.reset();
this._overviewTimelineView.reset();
this._clearTimelineNavigationItem.enabled = false;
this._exportButtonNavigationItem.enabled = false;
}
_recordingUnloaded(event)
{
console.assert(!this._updating);
WI.timelineManager.removeEventListener(null, null, this);
}
_timeRangeSelectionChanged(event)
{
console.assert(this.currentTimelineView);
if (!this.currentTimelineView)
return;
this._updateTimelineViewTimes(this.currentTimelineView);
let selectedPathComponent;
if (this._timelineOverview.timelineRuler.entireRangeSelected)
selectedPathComponent = this._entireRecordingPathComponent;
else {
let timelineRange = this._timelineSelectionPathComponent.representedObject;
timelineRange.startValue = this.currentTimelineView.startTime;
timelineRange.endValue = this.currentTimelineView.endTime;
if (!(this.currentTimelineView instanceof WI.RenderingFrameTimelineView)) {
timelineRange.startValue -= this.currentTimelineView.zeroTime;
timelineRange.endValue -= this.currentTimelineView.zeroTime;
}
this._updateTimeRangePathComponents();
selectedPathComponent = this._timelineSelectionPathComponent;
}
if (this._selectedTimeRangePathComponent !== selectedPathComponent) {
this._selectedTimeRangePathComponent = selectedPathComponent;
this.dispatchEventToListeners(WI.ContentView.Event.SelectionPathComponentsDidChange);
}
}
_recordSelected(event)
{
let {record} = event.data;
this._selectRecordInTimelineView(record);
}
_timelineSelected()
{
let timeline = this._timelineOverview.selectedTimeline;
if (timeline)
this.showTimelineViewForTimeline(timeline);
else
this.showOverviewTimelineView();
}
_updateTimeRangePathComponents()
{
let timelineRange = this._timelineSelectionPathComponent.representedObject;
let startValue = timelineRange.startValue;
let endValue = timelineRange.endValue;
if (isNaN(startValue) || isNaN(endValue)) {
this._entireRecordingPathComponent.nextSibling = null;
return;
}
this._entireRecordingPathComponent.nextSibling = this._timelineSelectionPathComponent;
let displayName;
if (this._timelineOverview.viewMode === WI.TimelineOverview.ViewMode.Timelines) {
const higherResolution = true;
let selectionStart = Number.secondsToString(startValue, higherResolution);
let selectionEnd = Number.secondsToString(endValue, higherResolution);
const epsilon = 0.0001;
if (startValue < epsilon)
displayName = WI.UIString("%s \u2013 %s").format(selectionStart, selectionEnd);
else {
let duration = Number.secondsToString(endValue - startValue, higherResolution);
displayName = WI.UIString("%s \u2013 %s (%s)").format(selectionStart, selectionEnd, duration);
}
} else {
startValue += 1; // Convert index to frame number.
if (startValue === endValue)
displayName = WI.UIString("Frame %d").format(startValue);
else
displayName = WI.UIString("Frames %d \u2013 %d").format(startValue, endValue);
}
this._timelineSelectionPathComponent.displayName = displayName;
this._timelineSelectionPathComponent.title = displayName;
}
_createTimelineRangePathComponent(title)
{
let range = new WI.TimelineRange(NaN, NaN);
let pathComponent = new WI.HierarchicalPathComponent(title || enDash, "time-icon", range);
pathComponent.addEventListener(WI.HierarchicalPathComponent.Event.SiblingWasSelected, this._timeRangePathComponentSelected, this);
return pathComponent;
}
_updateTimelineViewTimes(timelineView)
{
let timelineRuler = this._timelineOverview.timelineRuler;
let entireRangeSelected = timelineRuler.entireRangeSelected;
let endTime = this._timelineOverview.selectionStartTime + this._timelineOverview.selectionDuration;
if (entireRangeSelected) {
if (timelineView instanceof WI.RenderingFrameTimelineView)
endTime = this._renderingFrameTimeline.records.length;
else if (timelineView instanceof WI.HeapAllocationsTimelineView) {
// Since heap snapshots can be added at any time, including when not actively recording,
// make sure to set the end time to an effectively infinite number so any new records
// that are added in the future aren't filtered out.
endTime = Number.MAX_VALUE;
} else {
// Clamp selection to the end of the recording (with padding),
// so graph views will show an auto-sized graph without a lot of
// empty space at the end.
endTime = isNaN(this._recording.endTime) ? this._recording.currentTime : this._recording.endTime;
endTime += timelineRuler.minimumSelectionDuration;
}
}
timelineView.startTime = this._timelineOverview.selectionStartTime;
timelineView.currentTime = this._currentTime;
timelineView.endTime = endTime;
}
_editingInstrumentsDidChange(event)
{
let editingInstruments = this._timelineOverview.editingInstruments;
this.element.classList.toggle(WI.TimelineOverview.EditInstrumentsStyleClassName, editingInstruments);
this._updateTimelineOverviewHeight();
}
_filterDidChange()
{
if (!this.currentTimelineView)
return;
this.currentTimelineView.updateFilter(this._filterBarNavigationItem.filterBar.filters);
}
_handleTimelineViewRecordFiltered(event)
{
if (event.target !== this.currentTimelineView)
return;
console.assert(this.currentTimelineView);
let timeline = this.currentTimelineView.representedObject;
if (!(timeline instanceof WI.Timeline))
return;
let record = event.data.record;
let filtered = event.data.filtered;
this._timelineOverview.recordWasFiltered(timeline, record, filtered);
}
_handleTimelineViewRecordSelected(event)
{
if (!this.visible)
return;
let {record} = event.data;
this._selectRecordInTimelineOverview(record);
this._selectRecordInTimelineView(record);
}
_selectRecordInTimelineOverview(record)
{
let timeline = this._recording.timelineForRecordType(record.type);
if (!timeline)
return;
this._timelineOverview.selectRecord(timeline, record);
}
_selectRecordInTimelineView(record)
{
for (let timelineView of this._timelineViewMap.values()) {
let recordMatchesTimeline = record && timelineView.representedObject.type === record.type;
if (recordMatchesTimeline && timelineView !== this.currentTimelineView)
this.showTimelineViewForTimeline(timelineView.representedObject);
if (!record || recordMatchesTimeline)
timelineView.selectRecord(record);
}
}
_handleTimelineViewScannerShow(event)
{
if (!this.visible)
return;
let {time} = event.data;
this._timelineOverview.showScanner(time);
}
_handleTimelineViewScannerHide(event)
{
if (!this.visible)
return;
this._timelineOverview.hideScanner();
}
_handleTimelineViewNeedsEntireSelectedRange(event)
{
if (!this.visible)
return;
this._timelineOverview.timelineRuler.selectEntireRange();
}
_updateProgressView()
{
let isCapturing = WI.timelineManager.isCapturing();
this._progressView.visible = isCapturing && this.currentTimelineView && !this.currentTimelineView.showsLiveRecordingData;
}
_updateFilterBar()
{
this._filterBarNavigationItem.hidden = !this.currentTimelineView || !this.currentTimelineView.showsFilterBar;
}
};