UNPKG

monaca-lib

Version:

Monaca cloud API bindings for JavaScript

1,126 lines (1,040 loc) 69.6 kB
/* * Copyright (C) 2013 Google Inc. All rights reserved. * Copyright (C) 2012 Intel 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 */ WebInspector.TimelineUIUtils = function() { } /** * @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.TimelineUIUtils._initEventStyles = function() { if (WebInspector.TimelineUIUtils._eventStylesMap) return WebInspector.TimelineUIUtils._eventStylesMap; var recordTypes = WebInspector.TimelineModel.RecordType; var categories = WebInspector.TimelineUIUtils.categories(); var eventStyles = {}; eventStyles[recordTypes.Task] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Task"), categories["other"]); eventStyles[recordTypes.Program] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Other"), categories["other"]); eventStyles[recordTypes.Animation] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Animation"), categories["rendering"]); 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.PaintImage] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Paint Image"), categories["painting"], true); 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("Rasterize 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.ParseAuthorStyleSheet] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Parse Author Style Sheet"), 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("Timestamp"), categories["scripting"]); eventStyles[recordTypes.ConsoleTime] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Console Time"), categories["scripting"]); 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"]); eventStyles[recordTypes.GPUTask] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("GPU"), categories["gpu"]); WebInspector.TimelineUIUtils._eventStylesMap = eventStyles; return eventStyles; } WebInspector.TimelineUIUtils._coalescableRecordTypes = {}; WebInspector.TimelineUIUtils._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.Layout] = 1; WebInspector.TimelineUIUtils._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.Paint] = 1; WebInspector.TimelineUIUtils._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.RasterTask] = 1; WebInspector.TimelineUIUtils._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.DecodeImage] = 1; WebInspector.TimelineUIUtils._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.ResizeImage] = 1; /** * @param {string} recordType * @return {boolean} */ WebInspector.TimelineUIUtils.isCoalescable = function(recordType) { return !!WebInspector.TimelineUIUtils._coalescableRecordTypes[recordType]; } /** * @param {!WebInspector.TimelineModel.Record} record * @param {!RegExp} regExp * @return {boolean} */ WebInspector.TimelineUIUtils.testContentMatching = function(record, regExp) { var traceEvent = record.traceEvent(); var title = WebInspector.TimelineUIUtils.eventStyle(traceEvent).title; var tokens = [title]; if (traceEvent.url) tokens.push(traceEvent.url); 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 {!WebInspector.TimelineModel.Record} record * @return {!WebInspector.TimelineCategory} */ WebInspector.TimelineUIUtils.categoryForRecord = function(record) { return WebInspector.TimelineUIUtils.eventStyle(record.traceEvent()).category; } /** * @param {!WebInspector.TracingModel.Event} event * @return {!{title: string, category: !WebInspector.TimelineCategory}} */ WebInspector.TimelineUIUtils.eventStyle = function(event) { var eventStyles = WebInspector.TimelineUIUtils._initEventStyles(); if (event.category === WebInspector.TracingModel.ConsoleEventCategory) return { title: event.name, category: WebInspector.TimelineUIUtils.categories()["scripting"] }; var result = eventStyles[event.name]; if (!result) { result = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Unknown: %s", event.name), WebInspector.TimelineUIUtils.categories()["other"], true); eventStyles[event.name] = result; } return result; } /** * @param {!WebInspector.TracingModel.Event} event * @return {string} */ WebInspector.TimelineUIUtils.eventTitle = function(event) { var title = WebInspector.TimelineUIUtils.eventStyle(event).title; if (event.category === WebInspector.TracingModel.ConsoleEventCategory) return title; if (event.name === WebInspector.TimelineModel.RecordType.TimeStamp) return WebInspector.UIString("%s: %s", title, event.args["data"]["message"]); if (event.name === WebInspector.TimelineModel.RecordType.Animation && event.args["data"] && event.args["data"]["name"]) return WebInspector.UIString("%s: %s", title, event.args["data"]["name"]); return title; } /** * @param {!WebInspector.TracingModel.Event} event * @return {boolean} */ WebInspector.TimelineUIUtils.isMarkerEvent = function(event) { var recordTypes = WebInspector.TimelineModel.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.Target} target * @return {?string} */ WebInspector.TimelineUIUtils.buildDetailsTextForTraceEvent = function(event, target) { var recordType = WebInspector.TimelineModel.RecordType; 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: detailsText = linkifyLocationAsText(eventData["scriptId"], eventData["scriptLine"], 0); break; case recordType.JSFrame: detailsText = WebInspector.beautifyFunctionName(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: detailsText = linkifyTopCallFrameAsText() || eventData["timerId"]; break; case recordType.RequestAnimationFrame: case recordType.CancelAnimationFrame: detailsText = linkifyTopCallFrameAsText() || eventData["id"]; break; case recordType.ParseHTML: case recordType.RecalculateStyles: detailsText = linkifyTopCallFrameAsText(); break; case recordType.EvaluateScript: var url = eventData["url"]; if (url) detailsText = url + ":" + eventData["lineNumber"]; 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.url; if (url) detailsText = WebInspector.displayNameForURL(url); break; case recordType.Animation: detailsText = eventData && eventData["name"]; break; default: if (event.category === WebInspector.TracingModel.ConsoleEventCategory) detailsText = null; else detailsText = linkifyTopCallFrameAsText(); break; } return detailsText; /** * @param {string} scriptId * @param {number} lineNumber * @param {number=} columnNumber * @return {?string} */ function linkifyLocationAsText(scriptId, lineNumber, columnNumber) { // FIXME(62725): stack trace line/column numbers are one-based. var rawLocation = target && !target.isDetached() && scriptId ? target.debuggerModel.createRawLocationByScriptId(scriptId, lineNumber - 1, (columnNumber || 1) - 1) : null; if (!rawLocation) return null; var uiLocation = WebInspector.debuggerWorkspaceBinding.rawLocationToUILocation(rawLocation); return uiLocation.toUIString(); } /** * @return {?string} */ function linkifyTopCallFrameAsText() { var stackTrace = event.stackTrace; if (!stackTrace) { var initiator = event.initiator; if (initiator) stackTrace = initiator.stackTrace; } if (!stackTrace || !stackTrace.length) return null; var callFrame = stackTrace[0]; return linkifyLocationAsText(callFrame.scriptId, callFrame.lineNumber, callFrame.columnNumber); } } /** * @param {!WebInspector.TracingModel.Event} event * @param {?WebInspector.Target} target * @param {!WebInspector.Linkifier} linkifier * @return {?Node} */ WebInspector.TimelineUIUtils.buildDetailsNodeForTraceEvent = function(event, target, linkifier) { var recordType = WebInspector.TimelineModel.RecordType; var details; var detailsText; var eventData = event.args["data"]; switch (event.name) { case recordType.GCEvent: case recordType.TimerFire: case recordType.FireAnimationFrame: case recordType.EventDispatch: case recordType.Paint: case recordType.PaintImage: case recordType.DecodeImage: case recordType.ResizeImage: case recordType.DecodeLazyPixelRef: case recordType.Animation: case recordType.XHRReadyStateChange: case recordType.XHRLoad: case recordType.ResourceSendRequest: case recordType.ResourceReceivedData: case recordType.ResourceReceiveResponse: case recordType.ResourceFinish: case recordType.EmbedderCallback: detailsText = WebInspector.TimelineUIUtils.buildDetailsTextForTraceEvent(event, target); break; case recordType.FunctionCall: details = linkifyLocation(eventData["scriptId"], eventData["scriptName"], eventData["scriptLine"], 0); break; case recordType.JSFrame: details = createElement("span"); details.createTextChild(WebInspector.beautifyFunctionName(eventData["functionName"])); var location = linkifyLocation(eventData["scriptId"], eventData["url"], eventData["lineNumber"], eventData["columnNumber"]); if (location) { details.createTextChild(" @ "); details.appendChild(location); } 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; default: if (event.category === WebInspector.TracingModel.ConsoleEventCategory) detailsText = null; else details = linkifyTopCallFrame(); break; } if (!details && detailsText) details = 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.TimelineModel} model * @param {!WebInspector.Linkifier} linkifier * @param {function(!DocumentFragment)} callback */ WebInspector.TimelineUIUtils.buildTraceEventDetails = function(event, model, linkifier, callback) { var target = model.target(); if (!target) { callbackWrapper(); return; } var relatedNode = null; var barrier = new CallbackBarrier(); if (!event.previewElement) { if (event.url) WebInspector.DOMPresentationUtils.buildImagePreviewContents(target, event.url, false, barrier.createCallback(saveImage)); else if (event.picture) WebInspector.TimelineUIUtils.buildPicturePreviewContent(event, target, barrier.createCallback(saveImage)); } if (event.backendNodeId) target.domModel.pushNodesByBackendIdsToFrontend([event.backendNodeId], barrier.createCallback(setRelatedNode)); if (event.invalidationTrackingEvents) WebInspector.TimelineUIUtils._pushInvalidationNodeIdsToFrontend(event, target, barrier.createCallback(updateInvalidationNodeIds)); 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]); } /** * @param {?Array.<!DOMAgent.NodeId>} frontendNodeIds * @param {?Array.<!DOMAgent.NodeId>} backendNodeIds */ function updateInvalidationNodeIds(frontendNodeIds, backendNodeIds) { if (!frontendNodeIds) return; if (frontendNodeIds.length !== backendNodeIds.length) { console.error("Did not resolve the correct number of invalidation node ids."); return; } var backendToFrontendNodeIdMap = {}; backendNodeIds.forEach(function(backendNodeId, index) { backendToFrontendNodeIdMap[backendNodeId] = frontendNodeIds[index]; }); if (event.nodeId) event.frontendNodeId = backendToFrontendNodeIdMap[event.nodeId]; event.invalidationTrackingEvents.forEach(function(invalidation) { if (invalidation.nodeId) invalidation.frontendNodeId = backendToFrontendNodeIdMap[invalidation.nodeId]; }); } function callbackWrapper() { callback(WebInspector.TimelineUIUtils._buildTraceEventDetailsSynchronously(event, model, linkifier, relatedNode)); } } /** * @param {!WebInspector.TracingModel.Event} event * @param {!WebInspector.TimelineModel} model * @param {!WebInspector.Linkifier} linkifier * @param {?WebInspector.DOMNode} relatedNode * @return {!DocumentFragment} */ WebInspector.TimelineUIUtils._buildTraceEventDetailsSynchronously = function(event, model, linkifier, relatedNode) { var fragment = createDocumentFragment(); var stats = {}; var recordTypes = WebInspector.TimelineModel.RecordType; // This message may vary per event.name; var relatedNodeLabel; var contentHelper = new WebInspector.TimelineDetailsContentHelper(model.target(), linkifier, true); contentHelper.appendTextRow(WebInspector.UIString("Type"), WebInspector.TimelineUIUtils.eventTitle(event)); contentHelper.appendTextRow(WebInspector.UIString("Total Time"), Number.millisToString(event.duration || 0, true)); contentHelper.appendTextRow(WebInspector.UIString("Self Time"), Number.millisToString(event.selfTime, true)); if (event.previewElement) contentHelper.appendElementRow(WebInspector.UIString("Preview"), event.previewElement); 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: 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: 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("Owner element"); if (event.url) contentHelper.appendElementRow(WebInspector.UIString("Image URL"), WebInspector.linkifyResourceAsNode(event.url)); break; case recordTypes.ParseAuthorStyleSheet: var url = eventData["styleSheetUrl"]; if (url) contentHelper.appendElementRow(WebInspector.UIString("Stylesheet URL"), WebInspector.linkifyResourceAsNode(url)); break; case recordTypes.RecalculateStyles: // We don't want to see default details. contentHelper.appendTextRow(WebInspector.UIString("Elements affected"), event.args["elementCount"]); 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")); relatedNodeLabel = WebInspector.UIString("Layout root"); break; case recordTypes.ConsoleTime: contentHelper.appendTextRow(WebInspector.UIString("Message"), event.name); 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; case recordTypes.Animation: if (event.phase === WebInspector.TracingModel.Phase.NestableAsyncInstant) contentHelper.appendTextRow(WebInspector.UIString("State"), eventData["state"]); break; default: var detailsNode = WebInspector.TimelineUIUtils.buildDetailsNodeForTraceEvent(event, model.target(), 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"]); var warning = event.warning; if (warning) { var div = createElement("div"); div.textContent = warning; contentHelper.appendElementRow(WebInspector.UIString("Warning"), div); } var hasChildren = WebInspector.TimelineUIUtils._aggregatedStatsForTraceEvent(stats, model, event); if (hasChildren) { var pieChart = WebInspector.TimelineUIUtils.generatePieChart(stats, WebInspector.TimelineUIUtils.eventStyle(event).category, event.selfTime); contentHelper.appendElementRow(WebInspector.UIString("Aggregated Time"), pieChart); } if (event.stackTrace || (event.initiator && event.initiator.stackTrace) || event.invalidationTrackingEvents) WebInspector.TimelineUIUtils._generateCauses(event, model.target(), contentHelper); fragment.appendChild(contentHelper.element); return fragment; } /** * @param {!WebInspector.TracingModel.Event} event * @param {?WebInspector.Target} target * @param {!WebInspector.TimelineDetailsContentHelper} contentHelper */ WebInspector.TimelineUIUtils._generateCauses = function(event, target, contentHelper) { var recordTypes = WebInspector.TimelineModel.RecordType; var callSiteStackLabel; var stackLabel; var initiator = event.initiator; switch (event.name) { case recordTypes.TimerFire: callSiteStackLabel = WebInspector.UIString("Timer installed"); break; case recordTypes.FireAnimationFrame: callSiteStackLabel = WebInspector.UIString("Animation frame requested"); break; case recordTypes.RecalculateStyles: stackLabel = WebInspector.UIString("Recalculation was forced"); break; case recordTypes.Layout: callSiteStackLabel = WebInspector.UIString("First layout invalidation"); stackLabel = WebInspector.UIString("Layout forced"); break; } // Direct cause. if (event.stackTrace) contentHelper.appendStackTrace(stackLabel || WebInspector.UIString("Stack trace"), event.stackTrace); // Indirect causes. if (event.invalidationTrackingEvents) { // Full invalidation tracking (experimental). WebInspector.TimelineUIUtils._generateInvalidations(event, target, contentHelper); } else if (initiator && initiator.stackTrace) { // Partial invalidation tracking. contentHelper.appendStackTrace(callSiteStackLabel || WebInspector.UIString("First invalidated"), initiator.stackTrace); } } /** * @param {!WebInspector.TracingModel.Event} event * @param {?WebInspector.Target} target * @param {!WebInspector.TimelineDetailsContentHelper} contentHelper */ WebInspector.TimelineUIUtils._generateInvalidations = function(event, target, contentHelper) { if (!event.invalidationTrackingEvents) return; var invalidations = {}; event.invalidationTrackingEvents.forEach(function(invalidation) { if (!invalidations[invalidation.type]) invalidations[invalidation.type] = [invalidation]; else invalidations[invalidation.type].push(invalidation); }); Object.keys(invalidations).forEach(function(type) { WebInspector.TimelineUIUtils._generateInvalidationsForType( type, target, invalidations[type], contentHelper); }); } /** * @param {string} type * @param {?WebInspector.Target} target * @param {!Array.<!WebInspector.InvalidationTrackingEvent>} invalidations * @param {!WebInspector.TimelineDetailsContentHelper} contentHelper */ WebInspector.TimelineUIUtils._generateInvalidationsForType = function(type, target, invalidations, contentHelper) { var title; switch (type) { case WebInspector.TimelineModel.RecordType.StyleRecalcInvalidationTracking: title = WebInspector.UIString("Style invalidations"); break; case WebInspector.TimelineModel.RecordType.LayoutInvalidationTracking: title = WebInspector.UIString("Layout invalidations"); break; default: title = WebInspector.UIString("Other invalidations"); break; } var detailsNode = createElementWithClass("div", "timeline-details-view-row"); var titleElement = detailsNode.createChild("span", "timeline-details-view-row-title"); titleElement.textContent = WebInspector.UIString("%s: ", title); var eventsList = detailsNode.createChild("div", "timeline-details-view-row-value"); var invalidationGroups = groupInvalidationsByCause(invalidations); invalidationGroups.forEach(function(group) { appendInvalidationGroup(eventsList, group); }); contentHelper.element.appendChild(detailsNode); /** * @param {!Array.<!WebInspector.InvalidationTrackingEvent>} invalidations */ function groupInvalidationsByCause(invalidations) { var causeToInvalidationMap = {}; for (var index = 0; index < invalidations.length; index++) { var invalidation = invalidations[index]; var causeKey = ""; if (invalidation.cause.reason) causeKey += invalidation.cause.reason + "."; if (invalidation.cause.stackTrace) { invalidation.cause.stackTrace.forEach(function(stackFrame) { causeKey += stackFrame["functionName"] + "."; causeKey += stackFrame["scriptId"] + "."; causeKey += stackFrame["url"] + "."; causeKey += stackFrame["lineNumber"] + "."; causeKey += stackFrame["columnNumber"] + "."; }); } if (causeToInvalidationMap[causeKey]) causeToInvalidationMap[causeKey].push(invalidation); else causeToInvalidationMap[causeKey] = [ invalidation ]; } return Object.values(causeToInvalidationMap); } /** * @param {!Element} parentElement * @param {!Array.<!WebInspector.InvalidationTrackingEvent>} invalidations */ function appendInvalidationGroup(parentElement, invalidations) { if (!target) return; var row = parentElement.createChild("div", "invalidations-group section"); var header = row.createChild("div", "header"); header.addEventListener("click", function() { toggleDetails(header, invalidations); }); var first = invalidations[0]; var reason = first.cause.reason; var topFrame = first.cause.stackTrace && first.cause.stackTrace[0]; if (reason) header.createTextChild(WebInspector.UIString("%s for ", reason)); else header.createTextChild(WebInspector.UIString("Unknown cause for ")); appendTruncatedNodeList(header, invalidations); if (topFrame && contentHelper.linkifier()) { header.createTextChild(WebInspector.UIString(". ")); var stack = header.createChild("span", "monospace"); stack.createChild("span").textContent = WebInspector.beautifyFunctionName(topFrame.functionName); stack.createChild("span").textContent = " @ "; stack.createChild("span").appendChild(contentHelper.linkifier().linkifyConsoleCallFrame(target, topFrame)); } } /** * @param {!WebInspector.InvalidationTrackingEvent} invalidation * @param {boolean} showUnknownNodes */ function createInvalidationNode(invalidation, showUnknownNodes) { var node = target.domModel.nodeForId(invalidation.frontendNodeId); if (node) return WebInspector.DOMPresentationUtils.linkifyNodeReference(node); if (invalidation.nodeName) { var nodeSpan = createElement("span"); nodeSpan.textContent = WebInspector.UIString("[ %s ]", invalidation.nodeName); return nodeSpan; } if (showUnknownNodes) { var nodeSpan = createElement("span"); return nodeSpan.createTextChild(WebInspector.UIString("[ unknown node ]")); } } /** * @param {!Element} parentElement * @param {!Array.<!WebInspector.InvalidationTrackingEvent>} invalidations */ function appendTruncatedNodeList(parentElement, invalidations) { var invalidationNodes = []; var invalidationNodeIdMap = {}; for (var i = 0; i < invalidations.length; i++) { var invalidation = invalidations[i]; var invalidationNode = createInvalidationNode(invalidation, false); if (invalidationNode && !invalidationNodeIdMap[invalidation.nodeId]) { invalidationNodes.push(invalidationNode); invalidationNodeIdMap[invalidation.nodeId] = true; } } if (invalidationNodes.length === 1) { parentElement.appendChild(invalidationNodes[0]); } else if (invalidationNodes.length === 2) { parentElement.appendChild(invalidationNodes[0]); parentElement.createTextChild(WebInspector.UIString(" and ")); parentElement.appendChild(invalidationNodes[1]); } else if (invalidationNodes.length >= 3) { parentElement.appendChild(invalidationNodes[0]); parentElement.createTextChild(WebInspector.UIString(", ")); parentElement.appendChild(invalidationNodes[1]); parentElement.createTextChild(WebInspector.UIString(", and %s others", invalidationNodes.length - 2)); } } /** * @param {!Element} header * @param {!Array.<!WebInspector.InvalidationTrackingEvent>} invalidations */ function toggleDetails(header, invalidations) { var wasExpanded = header.classList.contains("expanded"); header.classList.toggle("expanded", !wasExpanded); header.parentElement.classList.toggle("expanded", !wasExpanded); if (wasExpanded) { var content = header.nextElementSibling; if (content) content.remove(); } else { createInvalidationGroupDetails(header.parentElement, invalidations); } } /** * @param {!Element} parentElement * @param {!Array.<!WebInspector.InvalidationTrackingEvent>} invalidations */ function createInvalidationGroupDetails(parentElement, invalidations) { var content = parentElement.createChild("div", "content"); var first = invalidations[0]; if (first.cause.stackTrace) { var stack = content.createChild("div"); stack.createTextChild(WebInspector.UIString("Stack trace:")); contentHelper.createChildStackTraceElement(stack, first.cause.stackTrace); } content.createTextChild(invalidations.length > 1 ? WebInspector.UIString("Nodes:") : WebInspector.UIString("Node:")); var nodeList = content.createChild("div", "node-list timeline-details-view-row-stack-trace"); appendDetailedNodeList(nodeList, invalidations); } /** * @param {!Element} parentElement * @param {!Array.<!WebInspector.InvalidationTrackingEvent>} invalidations */ function appendDetailedNodeList(parentElement, invalidations) { var firstNode = true; for (var i = 0; i < invalidations.length; i++) { var invalidation = invalidations[i]; var invalidationNode = createInvalidationNode(invalidation, true); if (invalidationNode) { if (!firstNode) parentElement.createTextChild(WebInspector.UIString(", ")); firstNode = false; parentElement.appendChild(invalidationNode); var extraData = invalidation.extraData ? ", " + invalidation.extraData : ""; if (invalidation.changedId) { parentElement.createTextChild(WebInspector.UIString("(changed id to \"%s\"%s)", invalidation.changedId, extraData)); } else if (invalidation.changedClass) { parentElement.createTextChild(WebInspector.UIString("(changed class to \"%s\"%s)", invalidation.changedClass, extraData)); } else if (invalidation.changedAttribute) { parentElement.createTextChild(WebInspector.UIString("(changed attribute to \"%s\"%s)", invalidation.changedAttribute, extraData)); } else if (invalidation.changedPseudo) { parentElement.createTextChild(WebInspector.UIString("(changed pesudo to \"%s\"%s)", invalidation.changedPseudo, extraData)); } else if (invalidation.selectorPart) { parentElement.createTextChild(WebInspector.UIString("(changed \"%s\"%s)", invalidation.selectorPart, extraData)); } } } } } /** * @param {!WebInspector.TracingModel.Event} event * @param {!WebInspector.Target} target * @param {function(?Array.<number>, ?Array.<number>)} callback */ WebInspector.TimelineUIUtils._pushInvalidationNodeIdsToFrontend = function(event, target, callback) { var backendNodeIds = []; var dedupedNodeIds = {}; if (event.nodeId) { backendNodeIds.push(event.nodeId); dedupedNodeIds[event.nodeId] = true; } event.invalidationTrackingEvents.forEach(function(invalidation) { if (invalidation.nodeId && !dedupedNodeIds[invalidation.nodeId]) { backendNodeIds.push(invalidation.nodeId); dedupedNodeIds[invalidation.nodeId] = true; } }); target.domModel.pushNodesByBackendIdsToFrontend(backendNodeIds, function(frontendNodeIds) { callback(frontendNodeIds, backendNodeIds); }); } /** * @param {!Object} total * @param {!WebInspector.TimelineModel.Record} record */ WebInspector.TimelineUIUtils.aggregateTimeForRecord = function(total, record) { var traceEvent = record.traceEvent(); var model = record.timelineModel(); WebInspector.TimelineUIUtils._aggregatedStatsForTraceEvent(total, model, traceEvent); } /** * @param {!Object} total * @param {!WebInspector.TimelineModel} model * @param {!WebInspector.TracingModel.Event} event * @return {boolean} */ WebInspector.TimelineUIUtils._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); // Not a main thread event? if (index < 0) return false; 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 categoryName = WebInspector.TimelineUIUtils.eventStyle(nextEvent).category.name; total[categoryName] = (total[categoryName] || 0) + nextEvent.selfTime; } } if (WebInspector.TracingModel.isAsyncPhase(event.phase)) { if (event.endTime) { var aggregatedTotal = 0; for (var categoryName in total) aggregatedTotal += total[categoryName]; total["idle"] = Math.max(0, event.endTime - event.startTime - aggregatedTotal); } return false; } return hasChildren; } /** * @param {!WebInspector.TracingModel.Event} event * @param {!WebInspector.Target} target * @param {function(!Element=)} callback */ WebInspector.TimelineUIUtils.buildPicturePreviewContent = function(event, target, callback) { new WebInspector.LayerPaintEvent(event, target).loadSnapshot(onSnapshotLoaded); /** * @param {?Array.<number>} rect * @param {?WebInspector.PaintProfilerSnapshot} snapshot */ function onSnapshotLoaded(rect, 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 = createElement("div"); container.classList.add("image-preview-container", "vbox", "link"); var img = container.createChild("img"); img.src = imageURL; var paintProfilerButton = container.createChild("a"); paintProfilerButton.textContent = WebInspector.UIString("Paint Profiler"); container.addEventListener("click", showPaintProfiler, false); callback(container); } function showPaintProfiler() { WebInspector.TimelinePanel.instance().select(WebInspector.TimelineSelection.fromTraceEvent(event), WebInspector.TimelinePanel.DetailsTab.PaintProfiler); } } /** * @param {string} recordType * @param {?string} title * @param {number} position * @return {!Element} */ WebInspector.TimelineUIUtils.createEventDivider = function(recordType, title, position) { var eventDivider = createElement("div"); eventDivider.className = "resources-event-divider"; var recordTypes = WebInspector.TimelineModel.RecordType; if (recordType === recordTypes.MarkDOMContent) eventDivider.className += " resources-blue-divider"; else if (recordType === recordTypes.M