chrome-devtools-frontend
Version:
Chrome DevTools UI
126 lines (104 loc) • 4.01 kB
text/typescript
// Copyright 2015 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.
import * as TraceEngine from '../../models/trace/trace.js';
import * as Platform from '../platform/platform.js';
import {TracingModel, type ObjectSnapshot} from './TracingModel.js';
export class FilmStripModel {
#framesInternal: Frame[];
#zeroTimeInternal: number;
#spanTimeInternal: number;
constructor(tracingModel: TracingModel, zeroTime?: number) {
this.#framesInternal = [];
this.#zeroTimeInternal = 0;
this.#spanTimeInternal = 0;
this.reset(tracingModel, zeroTime);
}
reset(tracingModel: TracingModel, zeroTime?: number): void {
this.#zeroTimeInternal = zeroTime || tracingModel.minimumRecordTime();
this.#spanTimeInternal = tracingModel.maximumRecordTime() - this.#zeroTimeInternal;
this.#framesInternal = [];
const browserMain = TracingModel.browserMainThread(tracingModel);
if (!browserMain) {
return;
}
const events = browserMain.events();
for (let i = 0; i < events.length; ++i) {
const event = events[i];
if (event.startTime < this.#zeroTimeInternal) {
continue;
}
if (!event.hasCategory(DEVTOOLS_SCREENSHOT_CATEGORY)) {
continue;
}
if (event.name === TraceEvents.Screenshot) {
this.#framesInternal.push(Frame.fromSnapshot(this, (event as ObjectSnapshot), this.#framesInternal.length));
}
}
}
frames(): Frame[] {
return this.#framesInternal;
}
zeroTime(): number {
return this.#zeroTimeInternal;
}
spanTime(): number {
return this.#spanTimeInternal;
}
frameByTimestamp(searchTimestamp: number): Frame|null {
// We want to find the closest frame to the timestamp that happened BEFORE
// the timestamp. So to do that we walk from the end of the array of
// frames, looking for the first frame where its timestamp is less than the
// timestamp we are searching for. It is important we search from the end
// of the array of frames, otherwise we will simply return the first frame
// that happened before the timestamp, even if it is not the closest one.
const closestFrameIndexBeforeTimestamp =
Platform.ArrayUtilities.nearestIndexFromEnd(this.#framesInternal, frame => frame.timestamp < searchTimestamp);
if (closestFrameIndexBeforeTimestamp === null) {
return null;
}
return this.#framesInternal[closestFrameIndexBeforeTimestamp];
}
}
const DEVTOOLS_SCREENSHOT_CATEGORY = 'disabled-by-default-devtools.screenshot';
const TraceEvents = {
Screenshot: 'Screenshot',
};
export class Frame {
readonly #modelInternal: FilmStripModel;
timestamp: number;
index: number;
#snapshot: ObjectSnapshot|null;
#traceEvent: TraceEngine.Types.TraceEvents.TraceEventSnapshot|null;
constructor(model: FilmStripModel, timestamp: number, index: number) {
this.#modelInternal = model;
this.timestamp = timestamp;
this.index = index;
this.#traceEvent = null;
this.#snapshot = null;
}
static fromSnapshot(model: FilmStripModel, snapshot: ObjectSnapshot, index: number): Frame {
const frame = new Frame(model, snapshot.startTime, index);
frame.#snapshot = snapshot;
return frame;
}
static fromTraceEvent(
model: FilmStripModel, snapshot: TraceEngine.Types.TraceEvents.TraceEventSnapshot, index: number): Frame {
const startTime = TraceEngine.Helpers.Timing.microSecondsToMilliseconds(snapshot.ts);
const frame = new Frame(model, startTime, index);
frame.#traceEvent = snapshot;
return frame;
}
model(): FilmStripModel {
return this.#modelInternal;
}
imageDataPromise(): Promise<string|null> {
if (this.#traceEvent) {
return Promise.resolve(this.#traceEvent.args.snapshot);
}
if (!this.#snapshot) {
return Promise.resolve(null);
}
return this.#snapshot.objectPromise() as Promise<string|null>;
}
}