Index: Source/devtools/front_end/timeline/TimelineUIUtilsImpl.js |
diff --git a/Source/devtools/front_end/timeline/TimelineUIUtilsImpl.js b/Source/devtools/front_end/timeline/TimelineUIUtilsImpl.js |
index d662b3b6ba61f653193020d102dd16d8bdcb585b..9a133a84b033c061d4ad0afa4edb5a6a1ca348d9 100644 |
--- a/Source/devtools/front_end/timeline/TimelineUIUtilsImpl.js |
+++ b/Source/devtools/front_end/timeline/TimelineUIUtilsImpl.js |
@@ -63,6 +63,29 @@ WebInspector.TimelineUIUtilsImpl.prototype = { |
} |
}, |
+ /** |
+ * @param {!WebInspector.TimelineModel.Record} record |
+ * @param {!WebInspector.Linkifier} linkifier |
+ * @param {boolean} loadedFromFile |
+ * @return {?Node} |
+ */ |
+ buildDetailsNode: function(record, linkifier, loadedFromFile) |
+ { |
+ return WebInspector.TimelineUIUtilsImpl.buildDetailsNode(record, linkifier, loadedFromFile); |
+ }, |
+ |
+ /** |
+ * @param {!WebInspector.TimelineModel.Record} record |
+ * @param {!WebInspector.TimelineModel} model |
+ * @param {!WebInspector.Linkifier} linkifier |
+ * @param {function(!DocumentFragment)} callback |
+ * @param {boolean} loadedFromFile |
+ */ |
+ generateDetailsContent: function(record, model, linkifier, callback, loadedFromFile) |
+ { |
+ WebInspector.TimelineUIUtilsImpl.generateDetailsContent(record, model, linkifier, callback, loadedFromFile); |
+ }, |
+ |
__proto__: WebInspector.TimelineUIUtils.prototype |
} |
@@ -73,3 +96,344 @@ WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes[WebInspector.TimelineMo |
WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.Rasterize] = 1; |
WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.DecodeImage] = 1; |
WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.ResizeImage] = 1; |
+ |
+ |
+/** |
+ * @param {!WebInspector.TimelineModel.Record} record |
+ * @param {!WebInspector.Linkifier} linkifier |
+ * @param {boolean} loadedFromFile |
+ * @return {?Node} |
+ */ |
+WebInspector.TimelineUIUtilsImpl.buildDetailsNode = function(record, linkifier, loadedFromFile) |
+{ |
+ var details; |
+ var detailsText; |
+ var recordData = record.data(); |
+ switch (record.type()) { |
+ case WebInspector.TimelineModel.RecordType.GCEvent: |
+ detailsText = WebInspector.UIString("%s collected", Number.bytesToString(recordData["usedHeapSizeDelta"])); |
+ break; |
+ case WebInspector.TimelineModel.RecordType.TimerFire: |
+ detailsText = recordData["timerId"]; |
+ break; |
+ case WebInspector.TimelineModel.RecordType.FunctionCall: |
+ details = linkifyLocation(recordData["scriptId"], recordData["scriptName"], recordData["scriptLine"], 0); |
+ break; |
+ case WebInspector.TimelineModel.RecordType.FireAnimationFrame: |
+ detailsText = recordData["id"]; |
+ break; |
+ case WebInspector.TimelineModel.RecordType.EventDispatch: |
+ detailsText = recordData ? recordData["type"] : null; |
+ break; |
+ case WebInspector.TimelineModel.RecordType.Paint: |
+ var width = WebInspector.TimelineUIUtils._quadWidth(recordData.clip); |
+ var height = WebInspector.TimelineUIUtils._quadHeight(recordData.clip); |
+ if (width && height) |
+ detailsText = WebInspector.UIString("%d\u2009\u00d7\u2009%d", width, height); |
+ break; |
+ case WebInspector.TimelineModel.RecordType.TimerInstall: |
+ case WebInspector.TimelineModel.RecordType.TimerRemove: |
+ details = linkifyTopCallFrame(); |
+ detailsText = recordData["timerId"]; |
+ break; |
+ case WebInspector.TimelineModel.RecordType.RequestAnimationFrame: |
+ case WebInspector.TimelineModel.RecordType.CancelAnimationFrame: |
+ details = linkifyTopCallFrame(); |
+ detailsText = recordData["id"]; |
+ break; |
+ case WebInspector.TimelineModel.RecordType.ParseHTML: |
+ case WebInspector.TimelineModel.RecordType.RecalculateStyles: |
+ details = linkifyTopCallFrame(); |
+ break; |
+ case WebInspector.TimelineModel.RecordType.EvaluateScript: |
+ var url = recordData["url"]; |
+ if (url) |
+ details = linkifyLocation("", url, recordData["lineNumber"], 0); |
+ break; |
+ case WebInspector.TimelineModel.RecordType.XHRReadyStateChange: |
+ case WebInspector.TimelineModel.RecordType.XHRLoad: |
+ case WebInspector.TimelineModel.RecordType.ResourceSendRequest: |
+ case WebInspector.TimelineModel.RecordType.DecodeImage: |
+ case WebInspector.TimelineModel.RecordType.ResizeImage: |
+ var url = recordData["url"]; |
+ if (url) |
+ detailsText = WebInspector.displayNameForURL(url); |
+ break; |
+ case WebInspector.TimelineModel.RecordType.ResourceReceivedData: |
+ case WebInspector.TimelineModel.RecordType.ResourceReceiveResponse: |
+ case WebInspector.TimelineModel.RecordType.ResourceFinish: |
+ var initiator = record.initiator(); |
+ if (initiator) { |
+ var url = initiator.data()["url"]; |
+ if (url) |
+ detailsText = WebInspector.displayNameForURL(url); |
+ } |
+ break; |
+ case WebInspector.TimelineModel.RecordType.ConsoleTime: |
+ detailsText = recordData["message"]; |
+ break; |
+ case WebInspector.TimelineModel.RecordType.EmbedderCallback: |
+ detailsText = recordData["callbackName"]; |
+ break; |
+ default: |
+ details = linkifyTopCallFrame(); |
+ break; |
+ } |
+ |
+ if (!details && detailsText) |
+ details = document.createTextNode(detailsText); |
+ return details; |
+ |
+ /** |
+ * @param {string} scriptId |
+ * @param {string} url |
+ * @param {number} lineNumber |
+ * @param {number=} columnNumber |
+ */ |
+ function linkifyLocation(scriptId, url, lineNumber, columnNumber) |
+ { |
+ if (!loadedFromFile && scriptId !== "0") { |
+ var location = new WebInspector.DebuggerModel.Location( |
+ record.target(), |
+ scriptId, |
+ lineNumber - 1, |
+ (columnNumber || 1) - 1); |
+ return linkifier.linkifyRawLocation(location, "timeline-details"); |
+ } |
+ |
+ if (!url) |
+ return null; |
+ |
+ // FIXME(62725): stack trace line/column numbers are one-based. |
+ columnNumber = columnNumber ? columnNumber - 1 : 0; |
+ return linkifier.linkifyLocation(record.target(), url, lineNumber - 1, columnNumber, "timeline-details"); |
+ } |
+ |
+ /** |
+ * @param {!ConsoleAgent.CallFrame} callFrame |
+ */ |
+ function linkifyCallFrame(callFrame) |
+ { |
+ return linkifyLocation(callFrame.scriptId, callFrame.url, callFrame.lineNumber, callFrame.columnNumber); |
+ } |
+ |
+ /** |
+ * @return {?Element} |
+ */ |
+ function linkifyTopCallFrame() |
+ { |
+ if (record.stackTrace()) |
+ return linkifyCallFrame(record.stackTrace()[0]); |
+ if (record.callSiteStackTrace()) |
+ return linkifyCallFrame(record.callSiteStackTrace()[0]); |
+ return null; |
+ } |
+} |
+ |
+/** |
+ * @param {!WebInspector.TimelineModel.Record} record |
+ * @param {!WebInspector.TimelineModel} model |
+ * @param {!WebInspector.Linkifier} linkifier |
+ * @param {function(!DocumentFragment)} callback |
+ * @param {boolean} loadedFromFile |
+ */ |
+WebInspector.TimelineUIUtilsImpl.generateDetailsContent = function(record, model, linkifier, callback, loadedFromFile) |
+{ |
+ var imageElement = /** @type {?Element} */ (record.getUserObject("TimelineUIUtils::preview-element") || null); |
+ var relatedNode = null; |
+ var recordData = record.data(); |
+ var barrier = new CallbackBarrier(); |
+ if (!imageElement && WebInspector.TimelineUIUtils.needsPreviewElement(record.type())) |
+ WebInspector.DOMPresentationUtils.buildImagePreviewContents(record.target(), recordData["url"], false, barrier.createCallback(saveImage)); |
+ if (recordData["backendNodeId"]) |
+ record.target().domModel.pushNodesByBackendIdsToFrontend([recordData["backendNodeId"]], barrier.createCallback(setRelatedNode)); |
+ barrier.callWhenDone(callbackWrapper); |
+ |
+ /** |
+ * @param {!Element=} element |
+ */ |
+ function saveImage(element) |
+ { |
+ imageElement = element || null; |
+ record.setUserObject("TimelineUIUtils::preview-element", element); |
+ } |
+ |
+ /** |
+ * @param {?Array.<!DOMAgent.NodeId>} nodeIds |
+ */ |
+ function setRelatedNode(nodeIds) |
+ { |
+ if (nodeIds) |
+ relatedNode = record.target().domModel.nodeForId(nodeIds[0]); |
+ } |
+ |
+ function callbackWrapper() |
+ { |
+ callback(WebInspector.TimelineUIUtilsImpl._generateDetailsContentSynchronously(record, model, linkifier, imageElement, relatedNode, loadedFromFile)); |
+ } |
+} |
+ |
+/** |
+ * @param {!WebInspector.TimelineModel.Record} record |
+ * @param {!WebInspector.TimelineModel} model |
+ * @param {!WebInspector.Linkifier} linkifier |
+ * @param {?Element} imagePreviewElement |
+ * @param {?WebInspector.DOMNode} relatedNode |
+ * @param {boolean} loadedFromFile |
+ * @return {!DocumentFragment} |
+ */ |
+WebInspector.TimelineUIUtilsImpl._generateDetailsContentSynchronously = function(record, model, linkifier, imagePreviewElement, relatedNode, loadedFromFile) |
+{ |
+ var fragment = document.createDocumentFragment(); |
+ if (record.children().length) |
+ fragment.appendChild(WebInspector.TimelineUIUtils.generatePieChart(record.aggregatedStats(), record.category(), record.selfTime())); |
+ else |
+ fragment.appendChild(WebInspector.TimelineUIUtils.generatePieChart(record.aggregatedStats())); |
+ |
+ const recordTypes = WebInspector.TimelineModel.RecordType; |
+ |
+ // The messages may vary per record.type(); |
+ var callSiteStackTraceLabel; |
+ var callStackLabel; |
+ var relatedNodeLabel; |
+ |
+ var contentHelper = new WebInspector.TimelineDetailsContentHelper(record.target(), linkifier, true); |
+ contentHelper.appendTextRow(WebInspector.UIString("Self Time"), Number.millisToString(record.selfTime(), true)); |
+ contentHelper.appendTextRow(WebInspector.UIString("Start Time"), Number.millisToString(record.startTime() - model.minimumRecordTime())); |
+ var recordData = record.data(); |
+ |
+ switch (record.type()) { |
+ case recordTypes.GCEvent: |
+ contentHelper.appendTextRow(WebInspector.UIString("Collected"), Number.bytesToString(recordData["usedHeapSizeDelta"])); |
+ break; |
+ case recordTypes.TimerFire: |
+ callSiteStackTraceLabel = WebInspector.UIString("Timer installed"); |
+ // Fall-through intended. |
+ |
+ case recordTypes.TimerInstall: |
+ case recordTypes.TimerRemove: |
+ contentHelper.appendTextRow(WebInspector.UIString("Timer ID"), recordData["timerId"]); |
+ if (record.type() === recordTypes.TimerInstall) { |
+ contentHelper.appendTextRow(WebInspector.UIString("Timeout"), Number.millisToString(recordData["timeout"])); |
+ contentHelper.appendTextRow(WebInspector.UIString("Repeats"), !recordData["singleShot"]); |
+ } |
+ break; |
+ case recordTypes.FireAnimationFrame: |
+ callSiteStackTraceLabel = WebInspector.UIString("Animation frame requested"); |
+ contentHelper.appendTextRow(WebInspector.UIString("Callback ID"), recordData["id"]); |
+ break; |
+ case recordTypes.FunctionCall: |
+ if (recordData["scriptName"]) |
+ contentHelper.appendLocationRow(WebInspector.UIString("Location"), recordData["scriptName"], recordData["scriptLine"]); |
+ break; |
+ case recordTypes.ResourceSendRequest: |
+ case recordTypes.ResourceReceiveResponse: |
+ case recordTypes.ResourceReceivedData: |
+ case recordTypes.ResourceFinish: |
+ var url; |
+ if (record.type() === recordTypes.ResourceSendRequest) |
+ url = recordData["url"]; |
+ else if (record.initiator()) |
+ url = record.initiator().data()["url"]; |
+ if (url) |
+ contentHelper.appendElementRow(WebInspector.UIString("Resource"), WebInspector.linkifyResourceAsNode(url)); |
+ if (imagePreviewElement) |
+ contentHelper.appendElementRow(WebInspector.UIString("Preview"), imagePreviewElement); |
+ if (recordData["requestMethod"]) |
+ contentHelper.appendTextRow(WebInspector.UIString("Request Method"), recordData["requestMethod"]); |
+ if (typeof recordData["statusCode"] === "number") |
+ contentHelper.appendTextRow(WebInspector.UIString("Status Code"), recordData["statusCode"]); |
+ if (recordData["mimeType"]) |
+ contentHelper.appendTextRow(WebInspector.UIString("MIME Type"), recordData["mimeType"]); |
+ if (recordData["encodedDataLength"]) |
+ contentHelper.appendTextRow(WebInspector.UIString("Encoded Data Length"), WebInspector.UIString("%d Bytes", recordData["encodedDataLength"])); |
+ break; |
+ case recordTypes.EvaluateScript: |
+ var url = recordData["url"]; |
+ if (url) |
+ contentHelper.appendLocationRow(WebInspector.UIString("Script"), url, recordData["lineNumber"]); |
+ break; |
+ case recordTypes.Paint: |
+ var clip = recordData["clip"]; |
+ contentHelper.appendTextRow(WebInspector.UIString("Location"), WebInspector.UIString("(%d, %d)", clip[0], clip[1])); |
+ var clipWidth = WebInspector.TimelineUIUtils._quadWidth(clip); |
+ var clipHeight = WebInspector.TimelineUIUtils._quadHeight(clip); |
+ contentHelper.appendTextRow(WebInspector.UIString("Dimensions"), WebInspector.UIString("%d × %d", clipWidth, clipHeight)); |
+ // Fall-through intended. |
+ |
+ case recordTypes.PaintSetup: |
+ case recordTypes.Rasterize: |
+ case recordTypes.ScrollLayer: |
+ relatedNodeLabel = WebInspector.UIString("Layer root"); |
+ break; |
+ case recordTypes.DecodeImage: |
+ case recordTypes.ResizeImage: |
+ relatedNodeLabel = WebInspector.UIString("Image element"); |
+ var url = recordData["url"]; |
+ if (url) |
+ contentHelper.appendElementRow(WebInspector.UIString("Image URL"), WebInspector.linkifyResourceAsNode(url)); |
+ break; |
+ case recordTypes.RecalculateStyles: // We don't want to see default details. |
+ if (recordData["elementCount"]) |
+ contentHelper.appendTextRow(WebInspector.UIString("Elements affected"), recordData["elementCount"]); |
+ callStackLabel = WebInspector.UIString("Styles recalculation forced"); |
+ break; |
+ case recordTypes.Layout: |
+ if (recordData["dirtyObjects"]) |
+ contentHelper.appendTextRow(WebInspector.UIString("Nodes that need layout"), recordData["dirtyObjects"]); |
+ if (recordData["totalObjects"]) |
+ contentHelper.appendTextRow(WebInspector.UIString("Layout tree size"), recordData["totalObjects"]); |
+ if (typeof recordData["partialLayout"] === "boolean") { |
+ contentHelper.appendTextRow(WebInspector.UIString("Layout scope"), |
+ recordData["partialLayout"] ? WebInspector.UIString("Partial") : WebInspector.UIString("Whole document")); |
+ } |
+ callSiteStackTraceLabel = WebInspector.UIString("Layout invalidated"); |
+ callStackLabel = WebInspector.UIString("Layout forced"); |
+ relatedNodeLabel = WebInspector.UIString("Layout root"); |
+ break; |
+ case recordTypes.ConsoleTime: |
+ contentHelper.appendTextRow(WebInspector.UIString("Message"), recordData["message"]); |
+ break; |
+ case recordTypes.WebSocketCreate: |
+ case recordTypes.WebSocketSendHandshakeRequest: |
+ case recordTypes.WebSocketReceiveHandshakeResponse: |
+ case recordTypes.WebSocketDestroy: |
+ var initiatorData = record.initiator() ? record.initiator().data() : recordData; |
+ if (typeof initiatorData["webSocketURL"] !== "undefined") |
+ contentHelper.appendTextRow(WebInspector.UIString("URL"), initiatorData["webSocketURL"]); |
+ if (typeof initiatorData["webSocketProtocol"] !== "undefined") |
+ contentHelper.appendTextRow(WebInspector.UIString("WebSocket Protocol"), initiatorData["webSocketProtocol"]); |
+ if (typeof recordData["message"] !== "undefined") |
+ contentHelper.appendTextRow(WebInspector.UIString("Message"), recordData["message"]); |
+ break; |
+ case recordTypes.EmbedderCallback: |
+ contentHelper.appendTextRow(WebInspector.UIString("Callback Function"), recordData["callbackName"]); |
+ break; |
+ default: |
+ var detailsNode = WebInspector.TimelineUIUtilsImpl.buildDetailsNode(record, linkifier, loadedFromFile); |
+ if (detailsNode) |
+ contentHelper.appendElementRow(WebInspector.UIString("Details"), detailsNode); |
+ break; |
+ } |
+ |
+ if (relatedNode) |
+ contentHelper.appendElementRow(relatedNodeLabel || WebInspector.UIString("Related node"), WebInspector.DOMPresentationUtils.linkifyNodeReference(relatedNode)); |
+ |
+ if (recordData["scriptName"] && record.type() !== recordTypes.FunctionCall) |
+ contentHelper.appendLocationRow(WebInspector.UIString("Function Call"), recordData["scriptName"], recordData["scriptLine"]); |
+ var callSiteStackTrace = record.callSiteStackTrace(); |
+ if (callSiteStackTrace) |
+ contentHelper.appendStackTrace(callSiteStackTraceLabel || WebInspector.UIString("Call Site stack"), callSiteStackTrace); |
+ var recordStackTrace = record.stackTrace(); |
+ if (recordStackTrace) |
+ contentHelper.appendStackTrace(callStackLabel || WebInspector.UIString("Call Stack"), recordStackTrace); |
+ |
+ if (record.warnings()) { |
+ var ul = document.createElement("ul"); |
+ for (var i = 0; i < record.warnings().length; ++i) |
+ ul.createChild("li").textContent = record.warnings()[i]; |
+ contentHelper.appendElementRow(WebInspector.UIString("Warning"), ul); |
+ } |
+ fragment.appendChild(contentHelper.element); |
+ return fragment; |
+} |