| Index: Source/devtools/front_end/timeline/TimelineFrameOverview.js
|
| diff --git a/Source/devtools/front_end/timeline/TimelineFrameOverview.js b/Source/devtools/front_end/timeline/TimelineFrameOverview.js
|
| deleted file mode 100644
|
| index 778450e862122acad5913a0c83579dc0079c02a6..0000000000000000000000000000000000000000
|
| --- a/Source/devtools/front_end/timeline/TimelineFrameOverview.js
|
| +++ /dev/null
|
| @@ -1,482 +0,0 @@
|
| -/*
|
| - * Copyright (C) 2013 Google 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:
|
| - *
|
| - * * Redistributions of source code must retain the above copyright
|
| - * notice, this list of conditions and the following disclaimer.
|
| - * * 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.
|
| - * * Neither the name of Google Inc. nor the names of its
|
| - * contributors may be used to endorse or promote products derived from
|
| - * this software without specific prior written permission.
|
| - *
|
| - * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT
|
| - * OWNER OR 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.
|
| - */
|
| -
|
| -/**
|
| - * @constructor
|
| - * @extends {WebInspector.TimelineOverviewBase}
|
| - * @param {!WebInspector.TimelineModel} model
|
| - * @param {!WebInspector.TimelineFrameModelBase} frameModel
|
| - */
|
| -WebInspector.TimelineFrameOverview = function(model, frameModel)
|
| -{
|
| - WebInspector.TimelineOverviewBase.call(this);
|
| - this.element.id = "timeline-overview-frames";
|
| - this._model = model;
|
| - this._frameModel = frameModel;
|
| - this.reset();
|
| -
|
| - this._outerPadding = 4 * window.devicePixelRatio;
|
| - this._maxInnerBarWidth = 10 * window.devicePixelRatio;
|
| - this._topPadding = 6 * window.devicePixelRatio;
|
| -
|
| - // The below two are really computed by update() -- but let's have something so that windowTimes() is happy.
|
| - this._actualPadding = 5 * window.devicePixelRatio;
|
| - this._actualOuterBarWidth = this._maxInnerBarWidth + this._actualPadding;
|
| -
|
| - this._fillStyles = {};
|
| - var categories = WebInspector.TimelineUIUtils.categories();
|
| - for (var category in categories)
|
| - this._fillStyles[category] = WebInspector.TimelineUIUtils.createFillStyleForCategory(this._context, this._maxInnerBarWidth, 0, categories[category]);
|
| -
|
| - this._frameTopShadeGradient = this._context.createLinearGradient(0, 0, 0, this._topPadding);
|
| - this._frameTopShadeGradient.addColorStop(0, "rgba(255, 255, 255, 0.9)");
|
| - this._frameTopShadeGradient.addColorStop(1, "rgba(255, 255, 255, 0.2)");
|
| -
|
| - this.element.addEventListener("mousemove", this._onMouseMove.bind(this), false);
|
| - this.element.addEventListener("mouseout", this._onMouseOut.bind(this), false);
|
| -}
|
| -
|
| -WebInspector.TimelineFrameOverview.Events = {
|
| - SelectionChanged: "SelectionChanged"
|
| -}
|
| -
|
| -WebInspector.TimelineFrameOverview.prototype = {
|
| - /**
|
| - * @override
|
| - */
|
| - reset: function()
|
| - {
|
| - /** @type {!Array<!{startTime:number, endTime:number}>} */
|
| - this._barTimes = [];
|
| - /** @type {!Array<!WebInspector.TimelineFrame>} */
|
| - this._visibleFrames = [];
|
| - this._selectedBarIndex = null;
|
| - this._activeBarIndex = null;
|
| - },
|
| -
|
| - /**
|
| - * @override
|
| - */
|
| - update: function()
|
| - {
|
| - this.resetCanvas();
|
| - this._barTimes = [];
|
| -
|
| - var minBarWidth = 4 * window.devicePixelRatio;
|
| - var frames = this._frameModel.frames();
|
| - var framesPerBar = Math.max(1, frames.length * minBarWidth / this._canvas.width);
|
| - this._visibleFrames = this._aggregateFrames(frames, framesPerBar);
|
| - this._scale = (this._canvas.height - this._topPadding) / this._computeTargetFrameLength(this._visibleFrames);
|
| - var maxPadding = 5 * window.devicePixelRatio;
|
| - this._actualOuterBarWidth = Math.min((this._canvas.width - 2 * this._outerPadding) / this._visibleFrames.length, this._maxInnerBarWidth + maxPadding);
|
| - this._actualPadding = Math.min(Math.floor(this._actualOuterBarWidth / 3), maxPadding);
|
| -
|
| - this._context.save();
|
| - for (var i = this._visibleFrames.length - 1; i >= 0; --i)
|
| - this._drawBar(i);
|
| - this._drawTopShadeGradient();
|
| - this._drawFPSMarks();
|
| - this._drawSelection();
|
| - this._context.restore();
|
| - },
|
| -
|
| - /**
|
| - * @param {?WebInspector.TimelineSelection} selection
|
| - */
|
| - select: function(selection)
|
| - {
|
| - var oldSelectionIndex = this._selectedBarIndex;
|
| - var frame = selection && selection.type() === WebInspector.TimelineSelection.Type.Frame ? /** @type {!WebInspector.TimelineFrame} */ (selection.object()) : null;
|
| - var index = frame ? this._visibleFrames.indexOf(frame) : -1;
|
| - this._selectedBarIndex = index >= 0 ? index : null;
|
| - if (this._selectedBarIndex === oldSelectionIndex)
|
| - return;
|
| - if (typeof oldSelectionIndex === "number")
|
| - this._redrawBar(oldSelectionIndex);
|
| - this._drawSelection();
|
| - },
|
| -
|
| - /**
|
| - * @override
|
| - * @param {!Event} event
|
| - * @return {boolean}
|
| - */
|
| - onClick: function(event)
|
| - {
|
| - var barIndex = this._screenPositionToBarIndex(event.clientX);
|
| - if (barIndex < 0 || barIndex >= this._visibleFrames.length)
|
| - return false;
|
| - var selection = WebInspector.TimelineSelection.fromFrame(this._visibleFrames[barIndex]);
|
| - this.dispatchEventToListeners(WebInspector.TimelineFrameOverview.Events.SelectionChanged, selection);
|
| - return true;
|
| - },
|
| -
|
| - /**
|
| - * @param {!Event} event
|
| - */
|
| - _onMouseMove: function(event)
|
| - {
|
| - var barIndex = this._screenPositionToBarIndex(event.clientX);
|
| - if (barIndex < 0 || barIndex >= this._visibleFrames.length)
|
| - barIndex = null;
|
| - this._setActiveBarIndex(barIndex);
|
| - },
|
| -
|
| - /**
|
| - * @param {!Event} event
|
| - */
|
| - _onMouseOut: function(event)
|
| - {
|
| - this._setActiveBarIndex(null);
|
| - },
|
| -
|
| - /**
|
| - * @param {?number} index
|
| - */
|
| - _setActiveBarIndex: function(index)
|
| - {
|
| - if (this._activeBarIndex === index)
|
| - return;
|
| - var oldActveBarIndex = this._activeBarIndex;
|
| - this._activeBarIndex = index;
|
| - if (typeof oldActveBarIndex === "number")
|
| - this._redrawBar(oldActveBarIndex);
|
| - if (typeof this._activeBarIndex === "number")
|
| - this._redrawBar(this._activeBarIndex);
|
| - },
|
| -
|
| - /**
|
| - * @param {number} index
|
| - */
|
| - _redrawBar: function(index)
|
| - {
|
| - this._context.save();
|
| - this._context.beginPath();
|
| - var left = this._barIndexToScreenPosition(index) - this._actualPadding;
|
| - var right = Math.ceil(left + this._actualOuterBarWidth);
|
| - this._context.rect(left, 0, right - left + 1, this._canvas.height);
|
| - this._context.fillStyle = "rgb(255, 255, 255)";
|
| - this._context.clip();
|
| - this._context.fill();
|
| - if (index > 0)
|
| - this._drawBar(index - 1);
|
| - if (index + 1 < this._visibleFrames.length)
|
| - this._drawBar(index + 1);
|
| - this._drawBar(index);
|
| - this._drawTopShadeGradient();
|
| - this._drawFPSMarks();
|
| - if (typeof this._selectedBarIndex === "number")
|
| - this._drawSelection();
|
| - this._context.restore();
|
| - },
|
| -
|
| - /**
|
| - * @param {!Array.<!WebInspector.TimelineFrame>} frames
|
| - * @param {number} framesPerBar
|
| - * @return {!Array.<!WebInspector.TimelineFrame>}
|
| - */
|
| - _aggregateFrames: function(frames, framesPerBar)
|
| - {
|
| - var visibleFrames = [];
|
| - for (var barIndex = 0, currentFrame = 0; currentFrame < frames.length; ++barIndex) {
|
| - var barStartTime = frames[currentFrame].startTime;
|
| - var longestFrame = null;
|
| - var longestDuration;
|
| -
|
| - for (var lastFrame = Math.min(Math.floor((barIndex + 1) * framesPerBar), frames.length);
|
| - currentFrame < lastFrame; ++currentFrame) {
|
| - var frame = frames[currentFrame];
|
| - var duration = frame.idle ? 0 : frame.duration; // Only consider idle frames if there are no regular frames.
|
| - if (!longestFrame || longestDuration < duration) {
|
| - longestFrame = frame;
|
| - longestDuration = duration;
|
| - }
|
| - }
|
| - var barEndTime = frames[currentFrame - 1].endTime;
|
| - if (longestFrame) {
|
| - visibleFrames.push(longestFrame);
|
| - this._barTimes.push({ startTime: barStartTime, endTime: barEndTime });
|
| - }
|
| - }
|
| - return visibleFrames;
|
| - },
|
| -
|
| - /**
|
| - * @param {!Array.<!WebInspector.TimelineFrame>} frames
|
| - * @return {number}
|
| - */
|
| - _computeTargetFrameLength: function(frames)
|
| - {
|
| - var targetFPS = 20;
|
| - var result = 1000.0 / targetFPS;
|
| - if (!frames.length)
|
| - return result;
|
| -
|
| - var durations = frames.select("duration");
|
| - var medianFrameLength = durations.qselect(Math.floor(durations.length / 2));
|
| -
|
| - // Optimize appearance for 30fps, but leave some space so it's evident when a frame overflows.
|
| - // However, if at least half frames won't fit at this scale, fall back to using autoscale.
|
| - if (result >= medianFrameLength)
|
| - return result;
|
| -
|
| - var maxFrameLength = Math.max.apply(Math, durations);
|
| - return Math.min(medianFrameLength * 2, maxFrameLength);
|
| - },
|
| -
|
| - /**
|
| - * @param {number} n
|
| - */
|
| - _barIndexToScreenPosition: function(n)
|
| - {
|
| - return this._outerPadding + this._actualOuterBarWidth * n;
|
| - },
|
| -
|
| - /**
|
| - * @param {number} clientX
|
| - */
|
| - _screenPositionToBarIndex: function(clientX)
|
| - {
|
| - var x = (clientX - this.element.totalOffsetLeft()) * window.devicePixelRatio;
|
| - return Math.floor((x - this._outerPadding) / this._actualOuterBarWidth);
|
| - },
|
| -
|
| - _drawTopShadeGradient: function()
|
| - {
|
| - this._context.fillStyle = this._frameTopShadeGradient;
|
| - this._context.fillRect(0, 0, this._canvas.width, this._topPadding);
|
| - },
|
| -
|
| - _drawFPSMarks: function()
|
| - {
|
| - var fpsMarks = [30, 60];
|
| -
|
| - this._context.save();
|
| - this._context.beginPath();
|
| - this._context.font = (10 * window.devicePixelRatio) + "px " + window.getComputedStyle(this.element, null).getPropertyValue("font-family");
|
| - this._context.textAlign = "right";
|
| - this._context.textBaseline = "alphabetic";
|
| -
|
| - var labelPadding = 4 * window.devicePixelRatio;
|
| - var baselineHeight = 3 * window.devicePixelRatio;
|
| - var lineHeight = 12 * window.devicePixelRatio;
|
| - var labelTopMargin = 0;
|
| - var labelOffsetY = 0; // Labels are going to be under their grid lines.
|
| -
|
| - for (var i = 0; i < fpsMarks.length; ++i) {
|
| - var fps = fpsMarks[i];
|
| - // Draw lines one pixel above they need to be, so 60pfs line does not cross most of the frames tops.
|
| - var y = this._canvas.height - Math.floor(1000.0 / fps * this._scale) - 0.5;
|
| - var label = WebInspector.UIString("%d\u2009fps", fps);
|
| - var labelWidth = this._context.measureText(label).width + 2 * labelPadding;
|
| - var labelX = this._canvas.width;
|
| -
|
| - if (!i && labelTopMargin < y - lineHeight)
|
| - labelOffsetY = -lineHeight; // Labels are going to be over their grid lines.
|
| - var labelY = y + labelOffsetY;
|
| - if (labelY < labelTopMargin || labelY + lineHeight > this._canvas.height)
|
| - break; // No space for the label, so no line as well.
|
| -
|
| - this._context.moveTo(0, y);
|
| - this._context.lineTo(this._canvas.width, y);
|
| -
|
| - this._context.fillStyle = "rgba(255, 255, 255, 0.5)";
|
| - this._context.fillRect(labelX - labelWidth, labelY, labelWidth, lineHeight);
|
| - this._context.fillStyle = "black";
|
| - this._context.fillText(label, labelX - labelPadding, labelY + lineHeight - baselineHeight);
|
| - labelTopMargin = labelY + lineHeight;
|
| - }
|
| - this._context.strokeStyle = "rgba(60, 60, 60, 0.4)";
|
| - this._context.stroke();
|
| - this._context.restore();
|
| - },
|
| -
|
| - /**
|
| - * @param {number} index
|
| - */
|
| - _drawBar: function(index)
|
| - {
|
| - var left = this._barIndexToScreenPosition(index);
|
| - var frame = this._visibleFrames[index];
|
| - var categories = Object.keys(WebInspector.TimelineUIUtils.categories());
|
| - var windowHeight = this._canvas.height;
|
| - var width = Math.floor(this._actualOuterBarWidth - this._actualPadding);
|
| -
|
| - var x = Math.floor(left) + 0.5;
|
| -
|
| - var totalCPUTime = frame.cpuTime;
|
| - var normalizedScale = this._scale;
|
| - if (totalCPUTime > frame.duration)
|
| - normalizedScale *= frame.duration / totalCPUTime;
|
| -
|
| - for (var i = 0, bottomOffset = windowHeight; i < categories.length; ++i) {
|
| - var category = categories[i];
|
| - var duration = frame.timeByCategory[category];
|
| - if (!duration)
|
| - continue;
|
| - var height = Math.round(duration * normalizedScale);
|
| - var y = Math.floor(bottomOffset - height) + 0.5;
|
| -
|
| - this._context.save();
|
| - this._context.translate(x, 0);
|
| - this._context.scale(width / this._maxInnerBarWidth, 1);
|
| - this._context.fillStyle = this._fillStyles[category];
|
| - this._context.fillRect(0, y, this._maxInnerBarWidth, Math.floor(height));
|
| - this._context.strokeStyle = WebInspector.TimelineUIUtils.categories()[category].borderColor;
|
| - this._context.beginPath();
|
| - this._context.moveTo(0, y);
|
| - this._context.lineTo(this._maxInnerBarWidth, y);
|
| - this._context.stroke();
|
| - this._context.restore();
|
| -
|
| - bottomOffset -= height;
|
| - }
|
| - // Skip outline for idle frames, unless frame is selected.
|
| - if (frame.idle && index !== this._activeBarIndex)
|
| - return;
|
| -
|
| - // Draw a contour for the total frame time.
|
| - var y0 = frame.idle ? bottomOffset + 0.5 : Math.floor(windowHeight - frame.duration * this._scale) + 0.5;
|
| - var y1 = windowHeight + 0.5;
|
| -
|
| - this._context.strokeStyle = index === this._activeBarIndex ? "rgba(0, 0, 0, 0.6)" : "rgba(90, 90, 90, 0.2)";
|
| - this._context.beginPath();
|
| - this._context.moveTo(x, y1);
|
| - this._context.lineTo(x, y0);
|
| - this._context.lineTo(x + width, y0);
|
| - this._context.lineTo(x + width, y1);
|
| - this._context.stroke();
|
| - },
|
| -
|
| - _drawSelection: function()
|
| - {
|
| - if (typeof this._selectedBarIndex !== "number")
|
| - return;
|
| - var left = this._barIndexToScreenPosition(this._selectedBarIndex);
|
| - var width = Math.floor(this._actualOuterBarWidth - this._actualPadding);
|
| - var triangleHeight = 4 * window.devicePixelRatio;
|
| - this._context.save();
|
| - this._context.beginPath();
|
| - this._context.moveTo(left, 0);
|
| - this._context.lineTo(left + width, 0);
|
| - this._context.lineTo(left + width / 2, triangleHeight);
|
| - this._context.closePath();
|
| - this._context.fillStyle = "black";
|
| - this._context.fill();
|
| - this._context.restore();
|
| - },
|
| -
|
| - /**
|
| - * @override
|
| - * @param {number} windowLeft
|
| - * @param {number} windowRight
|
| - * @return {!{startTime: number, endTime: number}}
|
| - */
|
| - windowTimes: function(windowLeft, windowRight)
|
| - {
|
| - if (!this._barTimes.length)
|
| - return WebInspector.TimelineOverviewBase.prototype.windowTimes.call(this, windowLeft, windowRight);
|
| - var windowSpan = this._canvas.width;
|
| - var leftOffset = windowLeft * windowSpan;
|
| - var rightOffset = windowRight * windowSpan;
|
| - var firstBar = Math.floor(Math.max(leftOffset - this._outerPadding + this._actualPadding, 0) / this._actualOuterBarWidth);
|
| - var lastBar = Math.min(Math.floor(Math.max(rightOffset - this._outerPadding, 0)/ this._actualOuterBarWidth), this._barTimes.length - 1);
|
| - if (firstBar >= this._barTimes.length)
|
| - return {startTime: Infinity, endTime: Infinity};
|
| -
|
| - var snapTolerancePixels = 3;
|
| - return {
|
| - startTime: leftOffset > snapTolerancePixels ? this._barTimes[firstBar].startTime : this._model.minimumRecordTime(),
|
| - endTime: (rightOffset + snapTolerancePixels > windowSpan) || (lastBar >= this._barTimes.length) ? this._model.maximumRecordTime() : this._barTimes[lastBar].endTime
|
| - };
|
| - },
|
| -
|
| - /**
|
| - * @override
|
| - * @param {number} startTime
|
| - * @param {number} endTime
|
| - * @return {!{left: number, right: number}}
|
| - */
|
| - windowBoundaries: function(startTime, endTime)
|
| - {
|
| - if (this._barTimes.length === 0)
|
| - return {left: 0, right: 1};
|
| - /**
|
| - * @param {number} time
|
| - * @param {!{startTime:number, endTime:number}} barTime
|
| - * @return {number}
|
| - */
|
| - function barStartComparator(time, barTime)
|
| - {
|
| - return time - barTime.startTime;
|
| - }
|
| - /**
|
| - * @param {number} time
|
| - * @param {!{startTime:number, endTime:number}} barTime
|
| - * @return {number}
|
| - */
|
| - function barEndComparator(time, barTime)
|
| - {
|
| - // We need a frame where time is in [barTime.startTime, barTime.endTime), so exclude exact matches against endTime.
|
| - if (time === barTime.endTime)
|
| - return 1;
|
| - return time - barTime.endTime;
|
| - }
|
| - return {
|
| - left: this._windowBoundaryFromTime(startTime, barEndComparator),
|
| - right: this._windowBoundaryFromTime(endTime, barStartComparator)
|
| - };
|
| - },
|
| -
|
| - /**
|
| - * @param {number} time
|
| - * @param {function(number, !{startTime:number, endTime:number}):number} comparator
|
| - */
|
| - _windowBoundaryFromTime: function(time, comparator)
|
| - {
|
| - if (time === Infinity)
|
| - return 1;
|
| - var index = this._firstBarAfter(time, comparator);
|
| - if (!index)
|
| - return 0;
|
| - return (this._barIndexToScreenPosition(index) - this._actualPadding / 2) / this._canvas.width;
|
| - },
|
| -
|
| - /**
|
| - * @param {number} time
|
| - * @param {function(number, {startTime:number, endTime:number}):number} comparator
|
| - */
|
| - _firstBarAfter: function(time, comparator)
|
| - {
|
| - return insertionIndexForObjectInListSortedByFunction(time, this._barTimes, comparator);
|
| - },
|
| -
|
| - __proto__: WebInspector.TimelineOverviewBase.prototype
|
| -}
|
|
|