| /* |
| * 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.View} |
| * @param {WebInspector.TimelineModel} model |
| */ |
| WebInspector.TimelineOverviewPane = function(model) |
| { |
| WebInspector.View.call(this); |
| this.element.id = "timeline-overview-panel"; |
| this.element.addStyleClass("hbox"); |
| |
| this._windowStartTime = 0; |
| this._windowEndTime = Infinity; |
| this._eventDividers = []; |
| |
| this._model = model; |
| |
| this._topPaneSidebarElement = document.createElement("div"); |
| this._topPaneSidebarElement.id = "timeline-overview-sidebar"; |
| |
| var overviewTreeElement = document.createElement("ol"); |
| overviewTreeElement.className = "sidebar-tree vbox"; |
| this._topPaneSidebarElement.appendChild(overviewTreeElement); |
| this.element.appendChild(this._topPaneSidebarElement); |
| |
| var topPaneSidebarTree = new TreeOutline(overviewTreeElement); |
| |
| this._overviewItems = {}; |
| this._overviewItems[WebInspector.TimelineOverviewPane.Mode.Events] = new WebInspector.SidebarTreeElement("timeline-overview-sidebar-events", |
| WebInspector.UIString("Events")); |
| this._overviewItems[WebInspector.TimelineOverviewPane.Mode.Frames] = new WebInspector.SidebarTreeElement("timeline-overview-sidebar-frames", |
| WebInspector.UIString("Frames")); |
| this._overviewItems[WebInspector.TimelineOverviewPane.Mode.Memory] = new WebInspector.SidebarTreeElement("timeline-overview-sidebar-memory", |
| WebInspector.UIString("Memory")); |
| |
| for (var mode in this._overviewItems) { |
| var item = this._overviewItems[mode]; |
| item.onselect = this.setMode.bind(this, mode); |
| topPaneSidebarTree.appendChild(item); |
| } |
| |
| this._overviewGrid = new WebInspector.OverviewGrid("timeline"); |
| |
| this.element.appendChild(this._overviewGrid.element); |
| |
| this._innerSetMode(WebInspector.TimelineOverviewPane.Mode.Events); |
| |
| var categories = WebInspector.TimelinePresentationModel.categories(); |
| for (var category in categories) |
| categories[category].addEventListener(WebInspector.TimelineCategory.Events.VisibilityChanged, this._onCategoryVisibilityChanged, this); |
| |
| this._overviewCalculator = new WebInspector.TimelineOverviewCalculator(); |
| |
| model.addEventListener(WebInspector.TimelineModel.Events.RecordAdded, this._onRecordAdded, this); |
| model.addEventListener(WebInspector.TimelineModel.Events.RecordsCleared, this._reset, this); |
| this._overviewGrid.addEventListener(WebInspector.OverviewGrid.Events.WindowChanged, this._onWindowChanged, this); |
| } |
| |
| WebInspector.TimelineOverviewPane.Mode = { |
| Events: "Events", |
| Frames: "Frames", |
| Memory: "Memory" |
| }; |
| |
| WebInspector.TimelineOverviewPane.Events = { |
| ModeChanged: "ModeChanged", |
| WindowChanged: "WindowChanged" |
| }; |
| |
| WebInspector.TimelineOverviewPane.prototype = { |
| wasShown: function() |
| { |
| this._update(); |
| }, |
| |
| onResize: function() |
| { |
| this._update(); |
| }, |
| |
| setMode: function(newMode) |
| { |
| if (this._currentMode === newMode) |
| return; |
| var windowTimes; |
| if (this._overviewControl) |
| windowTimes = this._overviewControl.windowTimes(this.windowLeft(), this.windowRight()); |
| this._innerSetMode(newMode); |
| this.dispatchEventToListeners(WebInspector.TimelineOverviewPane.Events.ModeChanged, this._currentMode); |
| if (windowTimes && windowTimes.startTime >= 0) |
| this.setWindowTimes(windowTimes.startTime, windowTimes.endTime); |
| this._update(); |
| }, |
| |
| _innerSetMode: function(newMode) |
| { |
| var windowTimes; |
| if (this._overviewControl) |
| this._overviewControl.detach(); |
| this._currentMode = newMode; |
| this._overviewControl = this._createOverviewControl(); |
| this._overviewControl.show(this._overviewGrid.element); |
| this._overviewItems[this._currentMode].revealAndSelect(false); |
| }, |
| |
| /** |
| * @return {WebInspector.TimelineOverviewBase|null} |
| */ |
| _createOverviewControl: function() |
| { |
| switch (this._currentMode) { |
| case WebInspector.TimelineOverviewPane.Mode.Events: |
| return new WebInspector.TimelineEventOverview(this._model); |
| case WebInspector.TimelineOverviewPane.Mode.Frames: |
| return new WebInspector.TimelineFrameOverview(this._model); |
| case WebInspector.TimelineOverviewPane.Mode.Memory: |
| return new WebInspector.TimelineMemoryOverview(this._model); |
| } |
| throw new Error("Invalid overview mode: " + this._currentMode); |
| }, |
| |
| _onCategoryVisibilityChanged: function(event) |
| { |
| this._overviewControl.categoryVisibilityChanged(); |
| }, |
| |
| _update: function() |
| { |
| delete this._refreshTimeout; |
| |
| this._updateWindow(); |
| this._overviewCalculator.setWindow(this._model.minimumRecordTime(), this._model.maximumRecordTime()); |
| this._overviewCalculator.setDisplayWindow(0, this._overviewGrid.clientWidth()); |
| |
| this._overviewControl.update(); |
| this._overviewGrid.updateDividers(this._overviewCalculator); |
| this._updateEventDividers(); |
| }, |
| |
| _updateEventDividers: function() |
| { |
| var records = this._eventDividers; |
| this._overviewGrid.removeEventDividers(); |
| var dividers = []; |
| for (var i = 0; i < records.length; ++i) { |
| var record = records[i]; |
| var positions = this._overviewCalculator.computeBarGraphPercentages(record); |
| var dividerPosition = Math.round(positions.start * 10); |
| if (dividers[dividerPosition]) |
| continue; |
| var divider = WebInspector.TimelinePresentationModel.createEventDivider(record.type); |
| divider.style.left = positions.start + "%"; |
| dividers[dividerPosition] = divider; |
| } |
| this._overviewGrid.addEventDividers(dividers); |
| }, |
| |
| /** |
| * @param {number} width |
| */ |
| sidebarResized: function(width) |
| { |
| this._topPaneSidebarElement.style.flexBasis = width + "px"; |
| this._update(); |
| }, |
| |
| /** |
| * @param {WebInspector.TimelineFrame} frame |
| */ |
| addFrame: function(frame) |
| { |
| this._overviewControl.addFrame(frame); |
| this._scheduleRefresh(); |
| }, |
| |
| /** |
| * @param {WebInspector.TimelineFrame} frame |
| */ |
| zoomToFrame: function(frame) |
| { |
| this.setWindowTimes(frame.startTime, frame.endTime); |
| this.dispatchEventToListeners(WebInspector.TimelineOverviewPane.Events.WindowChanged); |
| }, |
| |
| _onRecordAdded: function(event) |
| { |
| var record = event.data; |
| var eventDividers = this._eventDividers; |
| function addEventDividers(record) |
| { |
| if (WebInspector.TimelinePresentationModel.isEventDivider(record)) |
| eventDividers.push(record); |
| } |
| WebInspector.TimelinePresentationModel.forAllRecords([record], addEventDividers); |
| this._scheduleRefresh(); |
| }, |
| |
| _reset: function() |
| { |
| this._windowStartTime = 0; |
| this._windowEndTime = Infinity; |
| this._overviewCalculator.reset(); |
| this._overviewGrid.reset(); |
| this._overviewGrid.setResizeEnabled(false); |
| this._eventDividers = []; |
| this._overviewGrid.updateDividers(this._overviewCalculator); |
| this._overviewControl.reset(); |
| this._update(); |
| }, |
| |
| windowStartTime: function() |
| { |
| return this._windowStartTime || this._model.minimumRecordTime(); |
| }, |
| |
| windowEndTime: function() |
| { |
| return this._windowEndTime < Infinity ? this._windowEndTime : this._model.maximumRecordTime(); |
| }, |
| |
| windowLeft: function() |
| { |
| return this._overviewGrid.windowLeft(); |
| }, |
| |
| windowRight: function() |
| { |
| return this._overviewGrid.windowRight(); |
| }, |
| |
| _onWindowChanged: function() |
| { |
| if (this._ignoreWindowChangedEvent) |
| return; |
| var times = this._overviewControl.windowTimes(this.windowLeft(), this.windowRight()); |
| this._windowStartTime = times.startTime; |
| this._windowEndTime = times.endTime; |
| this.dispatchEventToListeners(WebInspector.TimelineOverviewPane.Events.WindowChanged); |
| }, |
| |
| /** |
| * @param {Number} startTime |
| * @param {Number} endTime |
| */ |
| setWindowTimes: function(startTime, endTime) |
| { |
| this._windowStartTime = startTime; |
| this._windowEndTime = endTime; |
| this._updateWindow(); |
| }, |
| |
| _updateWindow: function() |
| { |
| var windowBoundaries = this._overviewControl.windowBoundaries(this._windowStartTime, this._windowEndTime); |
| this._ignoreWindowChangedEvent = true; |
| this._overviewGrid.setWindow(windowBoundaries.left, windowBoundaries.right); |
| this._overviewGrid.setResizeEnabled(this._model.records.length); |
| this._ignoreWindowChangedEvent = false; |
| }, |
| |
| _scheduleRefresh: function() |
| { |
| if (this._refreshTimeout) |
| return; |
| if (!this.isShowing()) |
| return; |
| this._refreshTimeout = setTimeout(this._update.bind(this), 300); |
| }, |
| |
| __proto__: WebInspector.View.prototype |
| } |
| |
| /** |
| * @constructor |
| * @implements {WebInspector.TimelineGrid.Calculator} |
| */ |
| WebInspector.TimelineOverviewCalculator = function() |
| { |
| } |
| |
| WebInspector.TimelineOverviewCalculator.prototype = { |
| /** |
| * @param {number} time |
| */ |
| computePosition: function(time) |
| { |
| return (time - this._minimumBoundary) / this.boundarySpan() * this._workingArea + this.paddingLeft; |
| }, |
| |
| computeBarGraphPercentages: function(record) |
| { |
| var start = (WebInspector.TimelineModel.startTimeInSeconds(record) - this._minimumBoundary) / this.boundarySpan() * 100; |
| var end = (WebInspector.TimelineModel.endTimeInSeconds(record) - this._minimumBoundary) / this.boundarySpan() * 100; |
| return {start: start, end: end}; |
| }, |
| |
| /** |
| * @param {number=} minimum |
| * @param {number=} maximum |
| */ |
| setWindow: function(minimum, maximum) |
| { |
| this._minimumBoundary = minimum >= 0 ? minimum : undefined; |
| this._maximumBoundary = maximum >= 0 ? maximum : undefined; |
| }, |
| |
| /** |
| * @param {number} paddingLeft |
| * @param {number} clientWidth |
| */ |
| setDisplayWindow: function(paddingLeft, clientWidth) |
| { |
| this._workingArea = clientWidth - paddingLeft; |
| this.paddingLeft = paddingLeft; |
| }, |
| |
| reset: function() |
| { |
| this.setWindow(); |
| }, |
| |
| formatTime: function(value) |
| { |
| return Number.secondsToString(value); |
| }, |
| |
| maximumBoundary: function() |
| { |
| return this._maximumBoundary; |
| }, |
| |
| minimumBoundary: function() |
| { |
| return this._minimumBoundary; |
| }, |
| |
| zeroTime: function() |
| { |
| return this._minimumBoundary; |
| }, |
| |
| boundarySpan: function() |
| { |
| return this._maximumBoundary - this._minimumBoundary; |
| } |
| } |
| |
| /** |
| * @constructor |
| * @extends {WebInspector.View} |
| * @param {WebInspector.TimelineModel} model |
| */ |
| WebInspector.TimelineOverviewBase = function(model) |
| { |
| WebInspector.View.call(this); |
| this.element.classList.add("fill"); |
| |
| this._model = model; |
| this._canvas = this.element.createChild("canvas", "fill"); |
| this._context = this._canvas.getContext("2d"); |
| } |
| |
| WebInspector.TimelineOverviewBase.prototype = { |
| update: function() { }, |
| reset: function() { }, |
| |
| categoryVisibilityChanged: function() { }, |
| |
| /** |
| * @param {WebInspector.TimelineFrame} frame |
| */ |
| addFrame: function(frame) { }, |
| |
| /** |
| * @param {number} windowLeft |
| * @param {number} windowRight |
| */ |
| windowTimes: function(windowLeft, windowRight) |
| { |
| var absoluteMin = this._model.minimumRecordTime(); |
| var timeSpan = this._model.maximumRecordTime() - absoluteMin; |
| return { |
| startTime: absoluteMin + timeSpan * windowLeft, |
| endTime: absoluteMin + timeSpan * windowRight |
| }; |
| }, |
| |
| /** |
| * @param {number} startTime |
| * @param {number} endTime |
| */ |
| windowBoundaries: function(startTime, endTime) |
| { |
| var absoluteMin = this._model.minimumRecordTime(); |
| var timeSpan = this._model.maximumRecordTime() - absoluteMin; |
| var haveRecords = absoluteMin >= 0; |
| return { |
| left: haveRecords && startTime ? Math.min((startTime - absoluteMin) / timeSpan, 1) : 0, |
| right: haveRecords && endTime < Infinity ? (endTime - absoluteMin) / timeSpan : 1 |
| } |
| }, |
| |
| _resetCanvas: function() |
| { |
| this._canvas.width = this.element.clientWidth * window.devicePixelRatio; |
| this._canvas.height = this.element.clientHeight * window.devicePixelRatio; |
| }, |
| |
| __proto__: WebInspector.View.prototype |
| } |
| |
| /** |
| * @constructor |
| * @extends {WebInspector.TimelineOverviewBase} |
| * @param {WebInspector.TimelineModel} model |
| */ |
| WebInspector.TimelineMemoryOverview = function(model) |
| { |
| WebInspector.TimelineOverviewBase.call(this, model); |
| this.element.id = "timeline-overview-memory"; |
| |
| this._maxHeapSizeLabel = this.element.createChild("div", "max memory-graph-label"); |
| this._minHeapSizeLabel = this.element.createChild("div", "min memory-graph-label"); |
| } |
| |
| WebInspector.TimelineMemoryOverview.prototype = { |
| update: function() |
| { |
| this._resetCanvas(); |
| |
| var records = this._model.records; |
| if (!records.length) |
| return; |
| |
| const lowerOffset = 3; |
| var maxUsedHeapSize = 0; |
| var minUsedHeapSize = 100000000000; |
| var minTime = this._model.minimumRecordTime(); |
| var maxTime = this._model.maximumRecordTime(); |
| WebInspector.TimelinePresentationModel.forAllRecords(records, function(r) { |
| maxUsedHeapSize = Math.max(maxUsedHeapSize, r.usedHeapSize || maxUsedHeapSize); |
| minUsedHeapSize = Math.min(minUsedHeapSize, r.usedHeapSize || minUsedHeapSize); |
| }); |
| minUsedHeapSize = Math.min(minUsedHeapSize, maxUsedHeapSize); |
| |
| var width = this._canvas.width; |
| var height = this._canvas.height - lowerOffset; |
| var xFactor = width / (maxTime - minTime); |
| var yFactor = height / Math.max(maxUsedHeapSize - minUsedHeapSize, 1); |
| |
| var histogram = new Array(width); |
| WebInspector.TimelinePresentationModel.forAllRecords(records, function(r) { |
| if (!r.usedHeapSize) |
| return; |
| var x = Math.round((WebInspector.TimelineModel.endTimeInSeconds(r) - minTime) * xFactor); |
| var y = Math.round((r.usedHeapSize - minUsedHeapSize) * yFactor); |
| histogram[x] = Math.max(histogram[x] || 0, y); |
| }); |
| |
| height++; // +1 so that the border always fit into the canvas area. |
| |
| var y = 0; |
| var isFirstPoint = true; |
| var ctx = this._context; |
| ctx.beginPath(); |
| ctx.moveTo(0, this._canvas.height); |
| for (var x = 0; x < histogram.length; x++) { |
| if (typeof histogram[x] === "undefined") |
| continue; |
| if (isFirstPoint) { |
| isFirstPoint = false; |
| y = histogram[x]; |
| ctx.lineTo(0, height - y); |
| } |
| ctx.lineTo(x, height - y); |
| y = histogram[x]; |
| ctx.lineTo(x, height - y); |
| } |
| ctx.lineTo(width, height - y); |
| ctx.lineTo(width, this._canvas.height); |
| ctx.lineTo(0, this._canvas.height); |
| ctx.closePath(); |
| |
| ctx.lineWidth = 0.5; |
| ctx.strokeStyle = "rgba(20,0,0,0.8)"; |
| ctx.stroke(); |
| |
| ctx.fillStyle = "rgba(214,225,254, 0.8);"; |
| ctx.fill(); |
| |
| this._maxHeapSizeLabel.textContent = Number.bytesToString(maxUsedHeapSize); |
| this._minHeapSizeLabel.textContent = Number.bytesToString(minUsedHeapSize); |
| }, |
| |
| __proto__: WebInspector.TimelineOverviewBase.prototype |
| } |
| |
| /** |
| * @constructor |
| * @extends {WebInspector.TimelineOverviewBase} |
| * @param {WebInspector.TimelineModel} model |
| */ |
| WebInspector.TimelineEventOverview = function(model) |
| { |
| WebInspector.TimelineOverviewBase.call(this, model); |
| |
| this.element.id = "timeline-overview-events"; |
| |
| this._fillStyles = {}; |
| var categories = WebInspector.TimelinePresentationModel.categories(); |
| for (var category in categories) |
| this._fillStyles[category] = WebInspector.TimelinePresentationModel.createFillStyleForCategory(this._context, 0, WebInspector.TimelineEventOverview._stripGradientHeight, categories[category]); |
| |
| this._disabledCategoryFillStyle = WebInspector.TimelinePresentationModel.createFillStyle(this._context, 0, WebInspector.TimelineEventOverview._stripGradientHeight, |
| "rgb(218, 218, 218)", "rgb(170, 170, 170)", "rgb(143, 143, 143)"); |
| |
| this._disabledCategoryBorderStyle = "rgb(143, 143, 143)"; |
| } |
| |
| /** @const */ |
| WebInspector.TimelineEventOverview._numberOfStrips = 3; |
| |
| /** @const */ |
| WebInspector.TimelineEventOverview._stripGradientHeight = 120; |
| |
| WebInspector.TimelineEventOverview.prototype = { |
| update: function() |
| { |
| this._resetCanvas(); |
| |
| var stripHeight = Math.round(this._canvas.height / WebInspector.TimelineEventOverview._numberOfStrips); |
| var timeOffset = this._model.minimumRecordTime(); |
| var timeSpan = this._model.maximumRecordTime() - timeOffset; |
| var scale = this._canvas.width / timeSpan; |
| |
| var lastBarByGroup = []; |
| |
| this._context.fillStyle = "rgba(0, 0, 0, 0.05)"; |
| for (var i = 1; i < WebInspector.TimelineEventOverview._numberOfStrips; i += 2) |
| this._context.fillRect(0.5, i * stripHeight + 0.5, this._canvas.width, stripHeight); |
| |
| function appendRecord(record) |
| { |
| if (record.type === WebInspector.TimelineModel.RecordType.BeginFrame) |
| return; |
| var recordStart = Math.floor((WebInspector.TimelineModel.startTimeInSeconds(record) - timeOffset) * scale); |
| var recordEnd = Math.ceil((WebInspector.TimelineModel.endTimeInSeconds(record) - timeOffset) * scale); |
| var category = WebInspector.TimelinePresentationModel.categoryForRecord(record); |
| if (category.overviewStripGroupIndex < 0) |
| return; |
| var bar = lastBarByGroup[category.overviewStripGroupIndex]; |
| // This bar may be merged with previous -- so just adjust the previous bar. |
| const barsMergeThreshold = 2; |
| if (bar && bar.category === category && bar.end + barsMergeThreshold >= recordStart) { |
| if (recordEnd > bar.end) |
| bar.end = recordEnd; |
| return; |
| } |
| if (bar) |
| this._renderBar(bar.start, bar.end, stripHeight, bar.category); |
| lastBarByGroup[category.overviewStripGroupIndex] = { start: recordStart, end: recordEnd, category: category }; |
| } |
| WebInspector.TimelinePresentationModel.forAllRecords(this._model.records, appendRecord.bind(this)); |
| for (var i = 0; i < lastBarByGroup.length; ++i) { |
| if (lastBarByGroup[i]) |
| this._renderBar(lastBarByGroup[i].start, lastBarByGroup[i].end, stripHeight, lastBarByGroup[i].category); |
| } |
| }, |
| |
| categoryVisibilityChanged: function() |
| { |
| this.update(); |
| }, |
| |
| /** |
| * @param {number} begin |
| * @param {number} end |
| * @param {number} height |
| * @param {WebInspector.TimelineCategory} category |
| */ |
| _renderBar: function(begin, end, height, category) |
| { |
| const stripPadding = 4 * window.devicePixelRatio; |
| const innerStripHeight = height - 2 * stripPadding; |
| |
| var x = begin + 0.5; |
| var y = category.overviewStripGroupIndex * height + stripPadding + 0.5; |
| var width = Math.max(end - begin, 1); |
| |
| this._context.save(); |
| this._context.translate(x, y); |
| this._context.scale(1, innerStripHeight / WebInspector.TimelineEventOverview._stripGradientHeight); |
| this._context.fillStyle = category.hidden ? this._disabledCategoryFillStyle : this._fillStyles[category.name]; |
| this._context.fillRect(0, 0, width, WebInspector.TimelineEventOverview._stripGradientHeight); |
| this._context.strokeStyle = category.hidden ? this._disabledCategoryBorderStyle : category.borderColor; |
| this._context.strokeRect(0, 0, width, WebInspector.TimelineEventOverview._stripGradientHeight); |
| this._context.restore(); |
| }, |
| |
| __proto__: WebInspector.TimelineOverviewBase.prototype |
| } |
| |
| /** |
| * @constructor |
| * @extends {WebInspector.TimelineOverviewBase} |
| * @param {WebInspector.TimelineModel} model |
| */ |
| WebInspector.TimelineFrameOverview = function(model) |
| { |
| WebInspector.TimelineOverviewBase.call(this, model); |
| this.element.id = "timeline-overview-frames"; |
| this.reset(); |
| |
| this._outerPadding = 4 * window.devicePixelRatio; |
| this._maxInnerBarWidth = 10 * 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.TimelinePresentationModel.categories(); |
| for (var category in categories) |
| this._fillStyles[category] = WebInspector.TimelinePresentationModel.createFillStyleForCategory(this._context, this._maxInnerBarWidth, 0, categories[category]); |
| } |
| |
| WebInspector.TimelineFrameOverview.prototype = { |
| reset: function() |
| { |
| this._recordsPerBar = 1; |
| /** @type {!Array.<{startTime:number, endTime:number}>} */ |
| this._barTimes = []; |
| this._frames = []; |
| }, |
| |
| update: function() |
| { |
| this._resetCanvas(); |
| this._barTimes = []; |
| |
| const minBarWidth = 4 * window.devicePixelRatio; |
| var frameCount = this._frames.length; |
| var framesPerBar = Math.max(1, frameCount * minBarWidth / this._canvas.width); |
| var visibleFrames = this._aggregateFrames(this._frames, framesPerBar); |
| var windowHeight = this._canvas.height; |
| const paddingTop = 4 * window.devicePixelRatio; |
| var scale = (windowHeight - paddingTop) / this._computeTargetFrameLength(visibleFrames); |
| this._renderBars(visibleFrames, scale, windowHeight); |
| this._drawFPSMarks(scale, windowHeight); |
| }, |
| |
| /** |
| * @param {WebInspector.TimelineFrame} frame |
| */ |
| addFrame: function(frame) |
| { |
| this._frames.push(frame); |
| }, |
| |
| |
| /** |
| * @param {Array.<WebInspector.TimelineFrame>} frames |
| * @param {number} framesPerBar |
| * @return {Array.<WebInspector.TimelineFrame>} |
| */ |
| _aggregateFrames: function(frames, framesPerBar) |
| { |
| var visibleFrames = []; |
| for (var barNumber = 0, currentFrame = 0; currentFrame < frames.length; ++barNumber) { |
| var barStartTime = frames[currentFrame].startTime; |
| var longestFrame = null; |
| var longestDuration = 0; |
| |
| for (var lastFrame = Math.min(Math.floor((barNumber + 1) * framesPerBar), frames.length); |
| currentFrame < lastFrame; ++currentFrame) { |
| var duration = frames[currentFrame].duration; |
| if (!longestFrame || longestDuration < duration) { |
| longestFrame = frames[currentFrame]; |
| 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 durations = frames.select("duration"); |
| var medianFrameLength = durations.qselect(Math.floor(durations.length / 2)); |
| |
| // Optimize appearance for 30fps. However, if at least half frames won't fit at this scale, |
| // fall back to using autoscale. |
| const targetFPS = 30; |
| var result = 1.0 / targetFPS; |
| if (result >= medianFrameLength) |
| return result; |
| |
| var maxFrameLength = Math.max.apply(Math, durations); |
| return Math.min(medianFrameLength * 2, maxFrameLength); |
| }, |
| |
| /** |
| * @param {Array.<WebInspector.TimelineFrame>} frames |
| * @param {number} scale |
| * @param {number} windowHeight |
| */ |
| _renderBars: function(frames, scale, windowHeight) |
| { |
| const maxPadding = 5 * window.devicePixelRatio; |
| this._actualOuterBarWidth = Math.min((this._canvas.width - 2 * this._outerPadding) / frames.length, this._maxInnerBarWidth + maxPadding); |
| this._actualPadding = Math.min(Math.floor(this._actualOuterBarWidth / 3), maxPadding); |
| |
| var barWidth = this._actualOuterBarWidth - this._actualPadding; |
| for (var i = 0; i < frames.length; ++i) |
| this._renderBar(this._barNumberToScreenPosition(i), barWidth, windowHeight, frames[i], scale); |
| }, |
| |
| /** |
| * @param {number} n |
| */ |
| _barNumberToScreenPosition: function(n) |
| { |
| return this._outerPadding + this._actualOuterBarWidth * n; |
| }, |
| |
| /** |
| * @param {number} scale |
| * @param {number} height |
| */ |
| _drawFPSMarks: function(scale, height) |
| { |
| const 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"; |
| |
| const labelPadding = 4 * window.devicePixelRatio; |
| const 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 = height - Math.floor(1.0 / fps * 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 > 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(128, 128, 128, 0.5)"; |
| this._context.stroke(); |
| this._context.restore(); |
| }, |
| |
| /** |
| * @param {number} left |
| * @param {number} width |
| * @param {number} windowHeight |
| * @param {WebInspector.TimelineFrame} frame |
| * @param {number} scale |
| */ |
| _renderBar: function(left, width, windowHeight, frame, scale) |
| { |
| var categories = Object.keys(WebInspector.TimelinePresentationModel.categories()); |
| if (!categories.length) |
| return; |
| var x = Math.floor(left) + 0.5; |
| width = Math.floor(width); |
| |
| for (var i = 0, bottomOffset = windowHeight; i < categories.length; ++i) { |
| var category = categories[i]; |
| var duration = frame.timeByCategory[category]; |
| |
| if (!duration) |
| continue; |
| var height = duration * scale; |
| 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.TimelinePresentationModel.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; |
| } |
| // Draw a contour for the total frame time. |
| var y0 = Math.floor(windowHeight - frame.duration * scale) + 0.5; |
| var y1 = windowHeight + 0.5; |
| |
| this._context.strokeStyle = "rgba(90, 90, 90, 0.3)"; |
| 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(); |
| }, |
| |
| /** |
| * @param {number} windowLeft |
| * @param {number} windowRight |
| */ |
| 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 - this._outerPadding + this._actualPadding; |
| var rightOffset = windowRight * windowSpan - this._outerPadding; |
| var firstBar = Math.floor(Math.max(leftOffset, 0) / this._actualOuterBarWidth); |
| var lastBar = Math.min(Math.floor(rightOffset / this._actualOuterBarWidth), this._barTimes.length - 1); |
| if (firstBar >= this._barTimes.length) |
| return {startTime: Infinity, endTime: Infinity}; |
| |
| const snapToRightTolerancePixels = 3; |
| return { |
| startTime: this._barTimes[firstBar].startTime, |
| endTime: (rightOffset + snapToRightTolerancePixels > windowSpan) || (lastBar >= this._barTimes.length) ? Infinity : this._barTimes[lastBar].endTime |
| } |
| }, |
| |
| /** |
| * @param {number} startTime |
| * @param {number} endTime |
| */ |
| windowBoundaries: function(startTime, endTime) |
| { |
| /** |
| * @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._barNumberToScreenPosition(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 |
| } |
| |
| /** |
| * @param {WebInspector.TimelineOverviewPane} pane |
| * @constructor |
| * @implements {WebInspector.TimelinePresentationModel.Filter} |
| */ |
| WebInspector.TimelineWindowFilter = function(pane) |
| { |
| this._pane = pane; |
| } |
| |
| WebInspector.TimelineWindowFilter.prototype = { |
| /** |
| * @param {!WebInspector.TimelinePresentationModel.Record} record |
| * @return {boolean} |
| */ |
| accept: function(record) |
| { |
| return record.lastChildEndTime >= this._pane._windowStartTime && record.startTime <= this._pane._windowEndTime; |
| } |
| } |