206 lines
7.1 KiB
JavaScript
206 lines
7.1 KiB
JavaScript
/*
|
|
* Copyright (C) 2013 Apple Inc. All rights reserved.
|
|
*
|
|
* 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.Timeline = class Timeline extends WI.Object
|
|
{
|
|
constructor(type)
|
|
{
|
|
super();
|
|
|
|
this._type = type;
|
|
|
|
this.reset(true);
|
|
}
|
|
|
|
// Static
|
|
|
|
static create(type)
|
|
{
|
|
if (type === WI.TimelineRecord.Type.Network)
|
|
return new WI.NetworkTimeline(type);
|
|
|
|
if (type === WI.TimelineRecord.Type.CPU)
|
|
return new WI.CPUTimeline(type);
|
|
|
|
if (type === WI.TimelineRecord.Type.Memory)
|
|
return new WI.MemoryTimeline(type);
|
|
|
|
if (type === WI.TimelineRecord.Type.Media)
|
|
return new WI.MediaTimeline(type);
|
|
|
|
return new WI.Timeline(type);
|
|
}
|
|
|
|
// Public
|
|
|
|
get type() { return this._type; }
|
|
get startTime() { return this._startTime; }
|
|
get endTime() { return this._endTime; }
|
|
get records() { return this._records; }
|
|
|
|
reset(suppressEvents)
|
|
{
|
|
this._records = [];
|
|
this._startTime = NaN;
|
|
this._endTime = NaN;
|
|
|
|
if (!suppressEvents) {
|
|
this.dispatchEventToListeners(WI.Timeline.Event.TimesUpdated);
|
|
this.dispatchEventToListeners(WI.Timeline.Event.Reset);
|
|
}
|
|
}
|
|
|
|
addRecord(record, options = {})
|
|
{
|
|
if (record.updatesDynamically)
|
|
record.addEventListener(WI.TimelineRecord.Event.Updated, this._recordUpdated, this);
|
|
|
|
// Because records can be nested, it is possible that outer records with an early start time
|
|
// may be completed and added to the Timeline after inner records with a later start time
|
|
// were already added. In most cases this is a small drift, so make an effort to still keep
|
|
// the list sorted. Do it now, when inserting, so if the timeline is visible it has the
|
|
// best chance of being as accurate as possible during a recording.
|
|
this._tryInsertingRecordInSortedOrder(record);
|
|
|
|
this._updateTimesIfNeeded(record);
|
|
|
|
this.dispatchEventToListeners(WI.Timeline.Event.RecordAdded, {record});
|
|
}
|
|
|
|
saveIdentityToCookie(cookie)
|
|
{
|
|
cookie[WI.Timeline.TimelineTypeCookieKey] = this._type;
|
|
}
|
|
|
|
refresh()
|
|
{
|
|
this.dispatchEventToListeners(WI.Timeline.Event.Refreshed);
|
|
}
|
|
|
|
closestRecordTo(timestamp)
|
|
{
|
|
let lowerIndex = this._records.lowerBound(timestamp, (time, record) => time - record.endTime);
|
|
|
|
let recordBefore = this._records[lowerIndex - 1];
|
|
let recordAfter = this._records[lowerIndex];
|
|
if (!recordBefore && !recordAfter)
|
|
return null;
|
|
if (!recordBefore && recordAfter)
|
|
return recordAfter;
|
|
if (!recordAfter && recordBefore)
|
|
return recordBefore;
|
|
|
|
let before = Math.abs(recordBefore.endTime - timestamp);
|
|
let after = Math.abs(recordAfter.startTime - timestamp);
|
|
return (before < after) ? recordBefore : recordAfter;
|
|
}
|
|
|
|
recordsInTimeRange(startTime, endTime, {includeRecordBeforeStart, includeRecordAfterEnd} = {})
|
|
{
|
|
let lowerIndex = this._records.lowerBound(startTime, (time, record) => time - record.endTime);
|
|
if (includeRecordBeforeStart && lowerIndex > 0) {
|
|
lowerIndex--;
|
|
|
|
// If the record right before is a child of the same type of record, then use the parent as the before index.
|
|
let recordBefore = this._records[lowerIndex];
|
|
if (recordBefore.parent && recordBefore.parent.type === recordBefore.type) {
|
|
lowerIndex--;
|
|
while (this._records[lowerIndex] !== recordBefore.parent)
|
|
lowerIndex--;
|
|
}
|
|
}
|
|
|
|
let upperIndex = this._records.upperBound(endTime, (time, record) => time - record.startTime);
|
|
if (includeRecordAfterEnd && upperIndex < this._records.length)
|
|
++upperIndex;
|
|
|
|
return this._records.slice(lowerIndex, upperIndex);
|
|
}
|
|
|
|
// Private
|
|
|
|
_updateTimesIfNeeded(record)
|
|
{
|
|
let changed = false;
|
|
|
|
// Some records adjust their start time / end time to values that may be before
|
|
// or after the bounds the recording actually ran. Use the unadjusted times for
|
|
// the Timeline's bounds. Otherwise we may extend the timeline graphs to a time
|
|
// that was conceptually before / after the user started / stopping recording.
|
|
let recordStartTime = record.unadjustedStartTime;
|
|
let recordEndTime = record.unadjustedEndTime;
|
|
|
|
if (isNaN(this._startTime) || recordStartTime < this._startTime) {
|
|
this._startTime = recordStartTime;
|
|
changed = true;
|
|
}
|
|
|
|
if (isNaN(this._endTime) || this._endTime < recordEndTime) {
|
|
this._endTime = recordEndTime;
|
|
changed = true;
|
|
}
|
|
|
|
if (changed)
|
|
this.dispatchEventToListeners(WI.Timeline.Event.TimesUpdated);
|
|
}
|
|
|
|
_recordUpdated(event)
|
|
{
|
|
this._updateTimesIfNeeded(event.target);
|
|
}
|
|
|
|
_tryInsertingRecordInSortedOrder(record)
|
|
{
|
|
// Fast case add to the end.
|
|
let lastValue = this._records.lastValue;
|
|
if (!lastValue || lastValue.startTime < record.startTime || record.updatesDynamically) {
|
|
this._records.push(record);
|
|
return;
|
|
}
|
|
|
|
// Slow case, try to insert in the last 20 records.
|
|
let start = this._records.length - 2;
|
|
let end = Math.max(this._records.length - 20, 0);
|
|
for (let i = start; i >= end; --i) {
|
|
if (this._records[i].startTime < record.startTime) {
|
|
this._records.insertAtIndex(record, i + 1);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Give up and add to the end.
|
|
this._records.push(record);
|
|
}
|
|
};
|
|
|
|
WI.Timeline.Event = {
|
|
Reset: "timeline-reset",
|
|
RecordAdded: "timeline-record-added",
|
|
TimesUpdated: "timeline-times-updated",
|
|
Refreshed: "timeline-refreshed",
|
|
};
|
|
|
|
WI.Timeline.TimelineTypeCookieKey = "timeline-type";
|