UNPKG

strong-arc

Version:

A visual suite for the StrongLoop API Platform

824 lines (760 loc) 35.8 kB
// Copyright 2014 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. /** * @constructor * @extends {WebInspector.TimelineUIUtils} */ WebInspector.TracingTimelineUIUtils = function() { WebInspector.TimelineUIUtils.call(this); } WebInspector.TracingTimelineUIUtils.prototype = { /** * @param {!WebInspector.TimelineModel.Record} record * @return {boolean} */ isBeginFrame: function(record) { return record.type() === WebInspector.TracingTimelineModel.RecordType.BeginFrame; }, /** * @param {!WebInspector.TimelineModel.Record} record * @return {boolean} */ isProgram: function(record) { return record.type() === WebInspector.TracingTimelineModel.RecordType.Program; }, /** * @param {string} recordType * @return {boolean} */ isCoalescable: function(recordType) { return !!WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[recordType]; }, /** * @param {!WebInspector.TimelineModel.Record} record * @return {boolean} */ isEventDivider: function(record) { return WebInspector.TracingTimelineUIUtils.isMarkerEvent(record.traceEvent()); }, /** * @param {!WebInspector.TimelineModel.Record} record * @return {?Object} */ countersForRecord: function(record) { return record.type() === WebInspector.TracingTimelineModel.RecordType.UpdateCounters ? record.data() : null; }, /** * @param {!WebInspector.TimelineModel.Record} record * @return {?Object} */ highlightQuadForRecord: function(record) { return record.traceEvent().highlightQuad || null; }, /** * @param {!WebInspector.TimelineModel.Record} record * @return {string} */ titleForRecord: function(record) { var event = record.traceEvent(); return WebInspector.TracingTimelineUIUtils.eventTitle(event, record.timelineModel()); }, /** * @param {!WebInspector.TimelineModel.Record} record * @return {!WebInspector.TimelineCategory} */ categoryForRecord: function(record) { return WebInspector.TracingTimelineUIUtils.styleForTraceEvent(record.traceEvent().name).category; }, /** * @param {!WebInspector.TimelineModel.Record} record * @param {!WebInspector.Linkifier} linkifier * @return {?Node} */ buildDetailsNode: function(record, linkifier) { return WebInspector.TracingTimelineUIUtils.buildDetailsNodeForTraceEvent(record.traceEvent(), linkifier); }, /** * @param {!WebInspector.TimelineModel.Record} record * @param {!WebInspector.TimelineModel} model * @param {!WebInspector.Linkifier} linkifier * @param {function(!DocumentFragment)} callback */ generateDetailsContent: function(record, model, linkifier, callback) { if (!(model instanceof WebInspector.TracingTimelineModel)) throw new Error("Illegal argument."); var tracingTimelineModel = /** @type {!WebInspector.TracingTimelineModel} */ (model); WebInspector.TracingTimelineUIUtils.buildTraceEventDetails(record.traceEvent(), tracingTimelineModel, linkifier, callback); }, /** * @return {!Element} */ createBeginFrameDivider: function() { return this.createEventDivider(WebInspector.TracingTimelineModel.RecordType.BeginFrame); }, /** * @param {string} recordType * @param {string=} title * @return {!Element} */ createEventDivider: function(recordType, title) { return WebInspector.TracingTimelineUIUtils._createEventDivider(recordType, title); }, /** * @param {!WebInspector.TimelineModel.Record} record * @param {!RegExp} regExp * @return {boolean} */ testContentMatching: function(record, regExp) { var traceEvent = record.traceEvent(); var title = WebInspector.TracingTimelineUIUtils.styleForTraceEvent(traceEvent.name).title; var tokens = [title]; for (var argName in traceEvent.args) { var argValue = traceEvent.args[argName]; for (var key in argValue) tokens.push(argValue[key]); } return regExp.test(tokens.join("|")); }, /** * @param {!Object} total * @param {!WebInspector.TimelineModel.Record} record */ aggregateTimeForRecord: function(total, record) { var traceEvent = record.traceEvent(); var model = record._model; WebInspector.TracingTimelineUIUtils._aggregatedStatsForTraceEvent(total, model, traceEvent); }, /** * @return {!WebInspector.TimelineModel.Filter} */ hiddenRecordsFilter: function() { return new WebInspector.TimelineRecordVisibleTypeFilter(WebInspector.TracingTimelineUIUtils._visibleTypes()); }, /** * @return {?WebInspector.TimelineModel.Filter} */ hiddenEmptyRecordsFilter: function() { var hiddenEmptyRecords = [WebInspector.TimelineModel.RecordType.EventDispatch]; return new WebInspector.TimelineRecordHiddenEmptyTypeFilter(hiddenEmptyRecords); }, __proto__: WebInspector.TimelineUIUtils.prototype } /** * @constructor * @param {string} title * @param {!WebInspector.TimelineCategory} category * @param {boolean=} hidden */ WebInspector.TimelineRecordStyle = function(title, category, hidden) { this.title = title; this.category = category; this.hidden = !!hidden; } /** * @return {!Object.<string, !WebInspector.TimelineRecordStyle>} */ WebInspector.TracingTimelineUIUtils._initEventStyles = function() { if (WebInspector.TracingTimelineUIUtils._eventStylesMap) return WebInspector.TracingTimelineUIUtils._eventStylesMap; var recordTypes = WebInspector.TracingTimelineModel.RecordType; var categories = WebInspector.TimelineUIUtils.categories(); var eventStyles = {}; eventStyles[recordTypes.Program] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Other"), categories["other"]); eventStyles[recordTypes.EventDispatch] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Event"), categories["scripting"]); eventStyles[recordTypes.RequestMainThreadFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Request Main Thread Frame"), categories["rendering"], true); eventStyles[recordTypes.BeginFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Frame Start"), categories["rendering"], true); eventStyles[recordTypes.BeginMainThreadFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Frame Start (main thread)"), categories["rendering"], true); eventStyles[recordTypes.DrawFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Draw Frame"), categories["rendering"], true); eventStyles[recordTypes.ScheduleStyleRecalculation] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Schedule Style Recalculation"), categories["rendering"], true); eventStyles[recordTypes.RecalculateStyles] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Recalculate Style"), categories["rendering"]); eventStyles[recordTypes.InvalidateLayout] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Invalidate Layout"), categories["rendering"], true); eventStyles[recordTypes.Layout] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Layout"), categories["rendering"]); eventStyles[recordTypes.PaintSetup] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Paint Setup"), categories["painting"]); eventStyles[recordTypes.UpdateLayer] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Update Layer"), categories["painting"], true); eventStyles[recordTypes.UpdateLayerTree] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Update Layer Tree"), categories["rendering"]); eventStyles[recordTypes.Paint] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Paint"), categories["painting"]); eventStyles[recordTypes.RasterTask] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Paint"), categories["painting"]); eventStyles[recordTypes.ScrollLayer] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Scroll"), categories["rendering"]); eventStyles[recordTypes.CompositeLayers] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Composite Layers"), categories["painting"]); eventStyles[recordTypes.ParseHTML] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Parse HTML"), categories["loading"]); eventStyles[recordTypes.TimerInstall] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Install Timer"), categories["scripting"]); eventStyles[recordTypes.TimerRemove] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Remove Timer"), categories["scripting"]); eventStyles[recordTypes.TimerFire] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Timer Fired"), categories["scripting"]); eventStyles[recordTypes.XHRReadyStateChange] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("XHR Ready State Change"), categories["scripting"]); eventStyles[recordTypes.XHRLoad] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("XHR Load"), categories["scripting"]); eventStyles[recordTypes.EvaluateScript] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Evaluate Script"), categories["scripting"]); eventStyles[recordTypes.MarkLoad] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Load event"), categories["scripting"], true); eventStyles[recordTypes.MarkDOMContent] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("DOMContentLoaded event"), categories["scripting"], true); eventStyles[recordTypes.MarkFirstPaint] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("First paint"), categories["painting"], true); eventStyles[recordTypes.TimeStamp] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Stamp"), categories["scripting"]); eventStyles[recordTypes.ConsoleTime] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Console Time"), categories["scripting"], true); eventStyles[recordTypes.ResourceSendRequest] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Send Request"), categories["loading"]); eventStyles[recordTypes.ResourceReceiveResponse] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Receive Response"), categories["loading"]); eventStyles[recordTypes.ResourceFinish] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Finish Loading"), categories["loading"]); eventStyles[recordTypes.ResourceReceivedData] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Receive Data"), categories["loading"]); eventStyles[recordTypes.FunctionCall] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Function Call"), categories["scripting"]); eventStyles[recordTypes.GCEvent] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("GC Event"), categories["scripting"]); eventStyles[recordTypes.JSFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("JS Frame"), categories["scripting"]); eventStyles[recordTypes.RequestAnimationFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Request Animation Frame"), categories["scripting"]); eventStyles[recordTypes.CancelAnimationFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Cancel Animation Frame"), categories["scripting"]); eventStyles[recordTypes.FireAnimationFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Animation Frame Fired"), categories["scripting"]); eventStyles[recordTypes.WebSocketCreate] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Create WebSocket"), categories["scripting"]); eventStyles[recordTypes.WebSocketSendHandshakeRequest] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Send WebSocket Handshake"), categories["scripting"]); eventStyles[recordTypes.WebSocketReceiveHandshakeResponse] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Receive WebSocket Handshake"), categories["scripting"]); eventStyles[recordTypes.WebSocketDestroy] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Destroy WebSocket"), categories["scripting"]); eventStyles[recordTypes.EmbedderCallback] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Embedder Callback"), categories["scripting"]); eventStyles[recordTypes.DecodeImage] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Image Decode"), categories["painting"]); eventStyles[recordTypes.ResizeImage] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Image Resize"), categories["painting"]); WebInspector.TracingTimelineUIUtils._eventStylesMap = eventStyles; return eventStyles; } WebInspector.TracingTimelineUIUtils._coalescableRecordTypes = {}; WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[WebInspector.TracingTimelineModel.RecordType.Layout] = 1; WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[WebInspector.TracingTimelineModel.RecordType.Paint] = 1; WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[WebInspector.TracingTimelineModel.RecordType.RasterTask] = 1; WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[WebInspector.TracingTimelineModel.RecordType.DecodeImage] = 1; WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[WebInspector.TracingTimelineModel.RecordType.ResizeImage] = 1; /** * @param {!WebInspector.TracingModel.Event} event * @return {!{title: string, category: !WebInspector.TimelineCategory}} */ WebInspector.TracingTimelineUIUtils.eventStyle = function(event) { return WebInspector.TracingTimelineUIUtils.styleForTraceEvent(event.name); } /** * @param {string} name * @return {!{title: string, category: !WebInspector.TimelineCategory}} */ WebInspector.TracingTimelineUIUtils.styleForTraceEvent = function(name) { var eventStyles = WebInspector.TracingTimelineUIUtils._initEventStyles(); var result = eventStyles[name]; if (!result) { result = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Unknown: %s", name), WebInspector.TimelineUIUtils.categories()["other"]); eventStyles[name] = result; } return result; } /** * @param {!WebInspector.TracingModel.Event} event * @return {string} */ WebInspector.TracingTimelineUIUtils.markerEventColor = function(event) { var red = "rgb(255, 0, 0)"; var blue = "rgb(0, 0, 255)"; var orange = "rgb(255, 178, 23)"; var green = "rgb(0, 130, 0)"; if (event.category === WebInspector.TracingModel.ConsoleEventCategory) return orange; var recordTypes = WebInspector.TracingTimelineModel.RecordType; var eventName = event.name; switch (eventName) { case recordTypes.MarkDOMContent: return blue; case recordTypes.MarkLoad: return red; case recordTypes.MarkFirstPaint: return green; case recordTypes.TimeStamp: return orange; } return green; } /** * @param {!WebInspector.TracingModel.Event} event * @param {!WebInspector.TimelineModel} model * @return {string} */ WebInspector.TracingTimelineUIUtils.eventTitle = function(event, model) { if (event.category === WebInspector.TracingModel.ConsoleEventCategory) { var titleTemplate = WebInspector.UIString(event.phase === WebInspector.TracingModel.Phase.AsyncBegin ? "Console Time: %s" : "Console Time End: %s"); return WebInspector.UIString(titleTemplate, event.name); } if (event.name === WebInspector.TracingTimelineModel.RecordType.TimeStamp) return event.args["data"]["message"]; var title = WebInspector.TracingTimelineUIUtils.eventStyle(event).title; if (WebInspector.TracingTimelineUIUtils.isMarkerEvent(event)) { var startTime = Number.millisToString(event.startTime - model.minimumRecordTime()); return WebInspector.UIString("%s at %s", title, startTime); } return title; } /** * @param {!WebInspector.TracingModel.Event} event * @return {boolean} */ WebInspector.TracingTimelineUIUtils.isMarkerEvent = function(event) { if (event.category === WebInspector.TracingModel.ConsoleEventCategory) return true; var recordTypes = WebInspector.TracingTimelineModel.RecordType; switch (event.name) { case recordTypes.TimeStamp: case recordTypes.MarkFirstPaint: return true; case recordTypes.MarkDOMContent: case recordTypes.MarkLoad: return event.args["data"]["isMainFrame"]; default: return false; } } /** * @param {!WebInspector.TracingModel.Event} event * @param {!WebInspector.Linkifier} linkifier * @return {?Node} */ WebInspector.TracingTimelineUIUtils.buildDetailsNodeForTraceEvent = function(event, linkifier) { var recordType = WebInspector.TracingTimelineModel.RecordType; var target = event.thread.target(); var details; var detailsText; var eventData = event.args["data"]; switch (event.name) { case recordType.GCEvent: var delta = event.args["usedHeapSizeBefore"] - event.args["usedHeapSizeAfter"]; detailsText = WebInspector.UIString("%s collected", Number.bytesToString(delta)); break; case recordType.TimerFire: detailsText = eventData["timerId"]; break; case recordType.FunctionCall: details = linkifyLocation(eventData["scriptId"], eventData["scriptName"], eventData["scriptLine"], 0); break; case recordType.JSFrame: details = linkifyLocation(eventData["scriptId"], eventData["url"], eventData["lineNumber"], eventData["columnNumber"]); detailsText = eventData["functionName"]; break; case recordType.FireAnimationFrame: detailsText = eventData["id"]; break; case recordType.EventDispatch: detailsText = eventData ? eventData["type"] : null; break; case recordType.Paint: var width = WebInspector.TimelineUIUtils.quadWidth(eventData.clip); var height = WebInspector.TimelineUIUtils.quadHeight(eventData.clip); if (width && height) detailsText = WebInspector.UIString("%d\u2009\u00d7\u2009%d", width, height); break; case recordType.TimerInstall: case recordType.TimerRemove: details = linkifyTopCallFrame(); detailsText = eventData["timerId"]; break; case recordType.RequestAnimationFrame: case recordType.CancelAnimationFrame: details = linkifyTopCallFrame(); detailsText = eventData["id"]; break; case recordType.ParseHTML: case recordType.RecalculateStyles: details = linkifyTopCallFrame(); break; case recordType.EvaluateScript: var url = eventData["url"]; if (url) details = linkifyLocation("", url, eventData["lineNumber"], 0); break; case recordType.XHRReadyStateChange: case recordType.XHRLoad: case recordType.ResourceSendRequest: var url = eventData["url"]; if (url) detailsText = WebInspector.displayNameForURL(url); break; case recordType.ResourceReceivedData: case recordType.ResourceReceiveResponse: case recordType.ResourceFinish: var initiator = event.initiator; if (initiator) { var url = initiator.args["data"]["url"]; if (url) detailsText = WebInspector.displayNameForURL(url); } break; case recordType.EmbedderCallback: detailsText = eventData["callbackName"]; break; case recordType.PaintImage: case recordType.DecodeImage: case recordType.ResizeImage: case recordType.DecodeLazyPixelRef: var url = event.imageURL; if (url) detailsText = WebInspector.displayNameForURL(url); break; default: if (event.category === WebInspector.TracingModel.ConsoleEventCategory) detailsText = event.name; else details = linkifyTopCallFrame(); break; } if (detailsText) { if (details) details.textContent = detailsText; else 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 (!url) return null; // FIXME(62725): stack trace line/column numbers are one-based. return linkifier.linkifyScriptLocation(target, scriptId, url, lineNumber - 1, (columnNumber ||1) - 1, "timeline-details"); } /** * @return {?Element} */ function linkifyTopCallFrame() { var stackTrace = event.stackTrace; if (!stackTrace) { var initiator = event.initiator; if (initiator) stackTrace = initiator.stackTrace; } if (!stackTrace || !stackTrace.length) return null; return linkifier.linkifyConsoleCallFrame(target, stackTrace[0], "timeline-details"); } } /** * @param {!WebInspector.TracingModel.Event} event * @param {!WebInspector.TracingTimelineModel} model * @param {!WebInspector.Linkifier} linkifier * @param {function(!DocumentFragment)} callback */ WebInspector.TracingTimelineUIUtils.buildTraceEventDetails = function(event, model, linkifier, callback) { var target = event.thread.target(); var relatedNode = null; var barrier = new CallbackBarrier(); if (!event.previewElement && target) { if (event.imageURL) WebInspector.DOMPresentationUtils.buildImagePreviewContents(target, event.imageURL, false, barrier.createCallback(saveImage)); else if (event.picture) WebInspector.TracingTimelineUIUtils.buildPicturePreviewContent(target, event.picture, barrier.createCallback(saveImage)); } if (event.backendNodeId && target) target.domModel.pushNodesByBackendIdsToFrontend([event.backendNodeId], barrier.createCallback(setRelatedNode)); barrier.callWhenDone(callbackWrapper); /** * @param {!Element=} element */ function saveImage(element) { event.previewElement = element || null; } /** * @param {?Array.<!DOMAgent.NodeId>} nodeIds */ function setRelatedNode(nodeIds) { if (nodeIds) relatedNode = target.domModel.nodeForId(nodeIds[0]); } function callbackWrapper() { callback(WebInspector.TracingTimelineUIUtils._buildTraceEventDetailsSynchronously(event, model, linkifier, relatedNode)); } } /** * @param {!WebInspector.TracingModel.Event} event * @param {!WebInspector.TracingTimelineModel} model * @param {!WebInspector.Linkifier} linkifier * @param {?WebInspector.DOMNode} relatedNode * @return {!DocumentFragment} */ WebInspector.TracingTimelineUIUtils._buildTraceEventDetailsSynchronously = function(event, model, linkifier, relatedNode) { var fragment = document.createDocumentFragment(); var stats = {}; var hasChildren = WebInspector.TracingTimelineUIUtils._aggregatedStatsForTraceEvent(stats, model, event); var pieChart = hasChildren ? WebInspector.TimelineUIUtils.generatePieChart(stats, WebInspector.TracingTimelineUIUtils.styleForTraceEvent(event.name).category, event.selfTime) : WebInspector.TimelineUIUtils.generatePieChart(stats); fragment.appendChild(pieChart); var recordTypes = WebInspector.TracingTimelineModel.RecordType; // The messages may vary per event.name; var callSiteStackTraceLabel; var callStackLabel; var relatedNodeLabel; var contentHelper = new WebInspector.TimelineDetailsContentHelper(event.thread.target(), linkifier, true); contentHelper.appendTextRow(WebInspector.UIString("Self Time"), Number.millisToString(event.selfTime, true)); contentHelper.appendTextRow(WebInspector.UIString("Start Time"), Number.millisToString((event.startTime - model.minimumRecordTime()))); var eventData = event.args["data"]; var initiator = event.initiator; switch (event.name) { case recordTypes.GCEvent: var delta = event.args["usedHeapSizeBefore"] - event.args["usedHeapSizeAfter"]; contentHelper.appendTextRow(WebInspector.UIString("Collected"), Number.bytesToString(delta)); break; case recordTypes.TimerFire: callSiteStackTraceLabel = WebInspector.UIString("Timer installed"); // Fall-through intended. case recordTypes.TimerInstall: case recordTypes.TimerRemove: contentHelper.appendTextRow(WebInspector.UIString("Timer ID"), eventData["timerId"]); if (event.name === recordTypes.TimerInstall) { contentHelper.appendTextRow(WebInspector.UIString("Timeout"), Number.millisToString(eventData["timeout"])); contentHelper.appendTextRow(WebInspector.UIString("Repeats"), !eventData["singleShot"]); } break; case recordTypes.FireAnimationFrame: callSiteStackTraceLabel = WebInspector.UIString("Animation frame requested"); contentHelper.appendTextRow(WebInspector.UIString("Callback ID"), eventData["id"]); break; case recordTypes.FunctionCall: if (eventData["scriptName"]) contentHelper.appendLocationRow(WebInspector.UIString("Location"), eventData["scriptName"], eventData["scriptLine"]); break; case recordTypes.ResourceSendRequest: case recordTypes.ResourceReceiveResponse: case recordTypes.ResourceReceivedData: case recordTypes.ResourceFinish: var url = (event.name === recordTypes.ResourceSendRequest) ? eventData["url"] : initiator.args["data"]["url"]; if (url) contentHelper.appendElementRow(WebInspector.UIString("Resource"), WebInspector.linkifyResourceAsNode(url)); if (eventData["requestMethod"]) contentHelper.appendTextRow(WebInspector.UIString("Request Method"), eventData["requestMethod"]); if (typeof eventData["statusCode"] === "number") contentHelper.appendTextRow(WebInspector.UIString("Status Code"), eventData["statusCode"]); if (eventData["mimeType"]) contentHelper.appendTextRow(WebInspector.UIString("MIME Type"), eventData["mimeType"]); if (eventData["encodedDataLength"]) contentHelper.appendTextRow(WebInspector.UIString("Encoded Data Length"), WebInspector.UIString("%d Bytes", eventData["encodedDataLength"])); break; case recordTypes.EvaluateScript: var url = eventData["url"]; if (url) contentHelper.appendLocationRow(WebInspector.UIString("Script"), url, eventData["lineNumber"]); break; case recordTypes.Paint: var clip = eventData["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.PaintImage: case recordTypes.DecodeLazyPixelRef: case recordTypes.DecodeImage: case recordTypes.ResizeImage: case recordTypes.DrawLazyPixelRef: relatedNodeLabel = WebInspector.UIString("Image element"); if (event.imageURL) contentHelper.appendElementRow(WebInspector.UIString("Image URL"), WebInspector.linkifyResourceAsNode(event.imageURL)); break; case recordTypes.RecalculateStyles: // We don't want to see default details. contentHelper.appendTextRow(WebInspector.UIString("Elements affected"), event.args["elementCount"]); callStackLabel = WebInspector.UIString("Styles recalculation forced"); break; case recordTypes.Layout: var beginData = event.args["beginData"]; contentHelper.appendTextRow(WebInspector.UIString("Nodes that need layout"), beginData["dirtyObjects"]); contentHelper.appendTextRow(WebInspector.UIString("Layout tree size"), beginData["totalObjects"]); contentHelper.appendTextRow(WebInspector.UIString("Layout scope"), beginData["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"), eventData["message"]); break; case recordTypes.WebSocketCreate: case recordTypes.WebSocketSendHandshakeRequest: case recordTypes.WebSocketReceiveHandshakeResponse: case recordTypes.WebSocketDestroy: var initiatorData = initiator ? initiator.args["data"] : eventData; 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 eventData["message"] !== "undefined") contentHelper.appendTextRow(WebInspector.UIString("Message"), eventData["message"]); break; case recordTypes.EmbedderCallback: contentHelper.appendTextRow(WebInspector.UIString("Callback Function"), eventData["callbackName"]); break; default: var detailsNode = WebInspector.TracingTimelineUIUtils.buildDetailsNodeForTraceEvent(event, linkifier); if (detailsNode) contentHelper.appendElementRow(WebInspector.UIString("Details"), detailsNode); break; } if (relatedNode) contentHelper.appendElementRow(relatedNodeLabel || WebInspector.UIString("Related node"), WebInspector.DOMPresentationUtils.linkifyNodeReference(relatedNode)); if (eventData && eventData["scriptName"] && event.name !== recordTypes.FunctionCall) contentHelper.appendLocationRow(WebInspector.UIString("Function Call"), eventData["scriptName"], eventData["scriptLine"]); if (initiator) { var callSiteStackTrace = initiator.stackTrace; if (callSiteStackTrace) contentHelper.appendStackTrace(callSiteStackTraceLabel || WebInspector.UIString("Call Site stack"), callSiteStackTrace); } var eventStackTrace = event.stackTrace; if (eventStackTrace) contentHelper.appendStackTrace(callStackLabel || WebInspector.UIString("Call Stack"), eventStackTrace); var warning = event.warning; if (warning) { var div = document.createElement("div"); div.textContent = warning; contentHelper.appendElementRow(WebInspector.UIString("Warning"), div); } if (event.previewElement) contentHelper.appendElementRow(WebInspector.UIString("Preview"), event.previewElement); fragment.appendChild(contentHelper.element); return fragment; } /** * @param {!Object} total * @param {!WebInspector.TracingTimelineModel} model * @param {!WebInspector.TracingModel.Event} event * @return {boolean} */ WebInspector.TracingTimelineUIUtils._aggregatedStatsForTraceEvent = function(total, model, event) { var events = model.inspectedTargetEvents(); /** * @param {number} startTime * @param {!WebInspector.TracingModel.Event} e * @return {number} */ function eventComparator(startTime, e) { return startTime - e.startTime; } var index = events.binaryIndexOf(event.startTime, eventComparator); var hasChildren = false; var endTime = event.endTime; if (endTime) { for (var i = index; i < events.length; i++) { var nextEvent = events[i]; if (nextEvent.startTime >= endTime) break; if (!nextEvent.selfTime) continue; if (nextEvent.thread !== event.thread) continue; if (i > index) hasChildren = true; var category = WebInspector.TracingTimelineUIUtils.styleForTraceEvent(nextEvent.name).category.name; total[category] = (total[category] || 0) + nextEvent.selfTime; } } return hasChildren; } /** * @param {!WebInspector.Target} target * @param {string} encodedPicture * @param {function(!Element=)} callback */ WebInspector.TracingTimelineUIUtils.buildPicturePreviewContent = function(target, encodedPicture, callback) { WebInspector.PaintProfilerSnapshot.load(target, encodedPicture, onSnapshotLoaded); /** * @param {?WebInspector.PaintProfilerSnapshot} snapshot */ function onSnapshotLoaded(snapshot) { if (!snapshot) { callback(); return; } snapshot.requestImage(null, null, 1, onGotImage); snapshot.dispose(); } /** * @param {string=} imageURL */ function onGotImage(imageURL) { if (!imageURL) { callback(); return; } var container = document.createElement("div"); container.className = "image-preview-container"; var img = container.createChild("img"); img.src = imageURL; callback(container); } } /** * @param {string} recordType * @param {string=} title * @return {!Element} */ WebInspector.TracingTimelineUIUtils._createEventDivider = function(recordType, title) { var eventDivider = document.createElement("div"); eventDivider.className = "resources-event-divider"; var recordTypes = WebInspector.TracingTimelineModel.RecordType; if (recordType === recordTypes.MarkDOMContent) eventDivider.className += " resources-blue-divider"; else if (recordType === recordTypes.MarkLoad) eventDivider.className += " resources-red-divider"; else if (recordType === recordTypes.MarkFirstPaint) eventDivider.className += " resources-green-divider"; else if (recordType === recordTypes.TimeStamp || recordType === recordTypes.ConsoleTime) eventDivider.className += " resources-orange-divider"; else if (recordType === recordTypes.BeginFrame) eventDivider.className += " timeline-frame-divider"; if (title) eventDivider.title = title; return eventDivider; } /** * @return {!Array.<string>} */ WebInspector.TracingTimelineUIUtils._visibleTypes = function() { var eventStyles = WebInspector.TracingTimelineUIUtils._initEventStyles(); var result = []; for (var name in eventStyles) { if (!eventStyles[name].hidden) result.push(name); } return result; } /** * @return {!WebInspector.TracingTimelineModel.Filter} */ WebInspector.TracingTimelineUIUtils.hiddenEventsFilter = function() { return new WebInspector.TracingTimelineModel.InclusiveEventNameFilter(WebInspector.TracingTimelineUIUtils._visibleTypes()); }