1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
7 * @extends {WebInspector.TimelineUIUtils}
9 WebInspector.TracingTimelineUIUtils = function()
11 WebInspector.TimelineUIUtils.call(this);
14 WebInspector.TracingTimelineUIUtils.prototype = {
16 * @param {!WebInspector.TimelineModel.Record} record
19 isBeginFrame: function(record)
21 return record.type() === WebInspector.TracingTimelineModel.RecordType.BeginFrame;
25 * @param {!WebInspector.TimelineModel.Record} record
28 isProgram: function(record)
30 return record.type() === WebInspector.TracingTimelineModel.RecordType.Program;
34 * @param {string} recordType
37 isCoalescable: function(recordType)
39 return !!WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[recordType];
43 * @param {!WebInspector.TimelineModel.Record} record
46 isEventDivider: function(record)
48 return WebInspector.TracingTimelineUIUtils.isMarkerEvent(record.traceEvent());
52 * @param {!WebInspector.TimelineModel.Record} record
55 countersForRecord: function(record)
57 return record.type() === WebInspector.TracingTimelineModel.RecordType.UpdateCounters ? record.data() : null;
61 * @param {!WebInspector.TimelineModel.Record} record
64 highlightQuadForRecord: function(record)
66 return record.traceEvent().highlightQuad || null;
70 * @param {!WebInspector.TimelineModel.Record} record
73 titleForRecord: function(record)
75 var event = record.traceEvent();
76 return WebInspector.TracingTimelineUIUtils.eventTitle(event, record.timelineModel());
80 * @param {!WebInspector.TimelineModel.Record} record
81 * @return {!WebInspector.TimelineCategory}
83 categoryForRecord: function(record)
85 return WebInspector.TracingTimelineUIUtils.eventStyle(record.traceEvent()).category;
89 * @param {!WebInspector.TimelineModel.Record} record
90 * @param {!WebInspector.Linkifier} linkifier
93 buildDetailsNode: function(record, linkifier)
95 return WebInspector.TracingTimelineUIUtils.buildDetailsNodeForTraceEvent(record.traceEvent(), linkifier);
99 * @param {!WebInspector.TimelineModel.Record} record
100 * @param {!WebInspector.TimelineModel} model
101 * @param {!WebInspector.Linkifier} linkifier
102 * @param {function(!DocumentFragment)} callback
104 generateDetailsContent: function(record, model, linkifier, callback)
106 if (!(model instanceof WebInspector.TracingTimelineModel))
107 throw new Error("Illegal argument.");
108 var tracingTimelineModel = /** @type {!WebInspector.TracingTimelineModel} */ (model);
109 WebInspector.TracingTimelineUIUtils.buildTraceEventDetails(record.traceEvent(), tracingTimelineModel, linkifier, callback);
115 createBeginFrameDivider: function()
117 return this.createEventDivider(WebInspector.TracingTimelineModel.RecordType.BeginFrame);
121 * @param {string} recordType
122 * @param {string=} title
125 createEventDivider: function(recordType, title)
127 return WebInspector.TracingTimelineUIUtils._createEventDivider(recordType, title);
131 * @param {!WebInspector.TimelineModel.Record} record
132 * @param {!RegExp} regExp
135 testContentMatching: function(record, regExp)
137 var traceEvent = record.traceEvent();
138 var title = WebInspector.TracingTimelineUIUtils.eventStyle(traceEvent).title;
139 var tokens = [title];
140 for (var argName in traceEvent.args) {
141 var argValue = traceEvent.args[argName];
142 for (var key in argValue)
143 tokens.push(argValue[key]);
145 return regExp.test(tokens.join("|"));
149 * @param {!Object} total
150 * @param {!WebInspector.TimelineModel.Record} record
152 aggregateTimeForRecord: function(total, record)
154 var traceEvent = record.traceEvent();
155 var model = record._model;
156 WebInspector.TracingTimelineUIUtils._aggregatedStatsForTraceEvent(total, model, traceEvent);
160 * @return {!WebInspector.TimelineModel.Filter}
162 hiddenRecordsFilter: function()
164 return new WebInspector.TimelineRecordVisibleTypeFilter(WebInspector.TracingTimelineUIUtils._visibleTypes());
168 * @return {?WebInspector.TimelineModel.Filter}
170 hiddenEmptyRecordsFilter: function()
172 var hiddenEmptyRecords = [WebInspector.TimelineModel.RecordType.EventDispatch];
173 return new WebInspector.TimelineRecordHiddenEmptyTypeFilter(hiddenEmptyRecords);
176 __proto__: WebInspector.TimelineUIUtils.prototype
181 * @param {string} title
182 * @param {!WebInspector.TimelineCategory} category
183 * @param {boolean=} hidden
185 WebInspector.TimelineRecordStyle = function(title, category, hidden)
188 this.category = category;
189 this.hidden = !!hidden;
193 * @return {!Object.<string, !WebInspector.TimelineRecordStyle>}
195 WebInspector.TracingTimelineUIUtils._initEventStyles = function()
197 if (WebInspector.TracingTimelineUIUtils._eventStylesMap)
198 return WebInspector.TracingTimelineUIUtils._eventStylesMap;
200 var recordTypes = WebInspector.TracingTimelineModel.RecordType;
201 var categories = WebInspector.TimelineUIUtils.categories();
203 var eventStyles = {};
204 eventStyles[recordTypes.Program] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Other"), categories["other"]);
205 eventStyles[recordTypes.EventDispatch] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Event"), categories["scripting"]);
206 eventStyles[recordTypes.RequestMainThreadFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Request Main Thread Frame"), categories["rendering"], true);
207 eventStyles[recordTypes.BeginFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Frame Start"), categories["rendering"], true);
208 eventStyles[recordTypes.BeginMainThreadFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Frame Start (main thread)"), categories["rendering"], true);
209 eventStyles[recordTypes.DrawFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Draw Frame"), categories["rendering"], true);
210 eventStyles[recordTypes.ScheduleStyleRecalculation] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Schedule Style Recalculation"), categories["rendering"], true);
211 eventStyles[recordTypes.RecalculateStyles] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Recalculate Style"), categories["rendering"]);
212 eventStyles[recordTypes.InvalidateLayout] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Invalidate Layout"), categories["rendering"], true);
213 eventStyles[recordTypes.Layout] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Layout"), categories["rendering"]);
214 eventStyles[recordTypes.PaintSetup] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Paint Setup"), categories["painting"]);
215 eventStyles[recordTypes.UpdateLayer] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Update Layer"), categories["painting"], true);
216 eventStyles[recordTypes.UpdateLayerTree] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Update Layer Tree"), categories["rendering"]);
217 eventStyles[recordTypes.Paint] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Paint"), categories["painting"]);
218 eventStyles[recordTypes.RasterTask] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Paint"), categories["painting"]);
219 eventStyles[recordTypes.ScrollLayer] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Scroll"), categories["rendering"]);
220 eventStyles[recordTypes.CompositeLayers] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Composite Layers"), categories["painting"]);
221 eventStyles[recordTypes.ParseHTML] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Parse HTML"), categories["loading"]);
222 eventStyles[recordTypes.TimerInstall] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Install Timer"), categories["scripting"]);
223 eventStyles[recordTypes.TimerRemove] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Remove Timer"), categories["scripting"]);
224 eventStyles[recordTypes.TimerFire] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Timer Fired"), categories["scripting"]);
225 eventStyles[recordTypes.XHRReadyStateChange] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("XHR Ready State Change"), categories["scripting"]);
226 eventStyles[recordTypes.XHRLoad] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("XHR Load"), categories["scripting"]);
227 eventStyles[recordTypes.EvaluateScript] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Evaluate Script"), categories["scripting"]);
228 eventStyles[recordTypes.MarkLoad] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Load event"), categories["scripting"], true);
229 eventStyles[recordTypes.MarkDOMContent] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("DOMContentLoaded event"), categories["scripting"], true);
230 eventStyles[recordTypes.MarkFirstPaint] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("First paint"), categories["painting"], true);
231 eventStyles[recordTypes.TimeStamp] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Timestamp"), categories["scripting"]);
232 eventStyles[recordTypes.ConsoleTime] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Console Time"), categories["scripting"]);
233 eventStyles[recordTypes.ResourceSendRequest] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Send Request"), categories["loading"]);
234 eventStyles[recordTypes.ResourceReceiveResponse] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Receive Response"), categories["loading"]);
235 eventStyles[recordTypes.ResourceFinish] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Finish Loading"), categories["loading"]);
236 eventStyles[recordTypes.ResourceReceivedData] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Receive Data"), categories["loading"]);
237 eventStyles[recordTypes.FunctionCall] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Function Call"), categories["scripting"]);
238 eventStyles[recordTypes.GCEvent] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("GC Event"), categories["scripting"]);
239 eventStyles[recordTypes.JSFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("JS Frame"), categories["scripting"]);
240 eventStyles[recordTypes.RequestAnimationFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Request Animation Frame"), categories["scripting"]);
241 eventStyles[recordTypes.CancelAnimationFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Cancel Animation Frame"), categories["scripting"]);
242 eventStyles[recordTypes.FireAnimationFrame] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Animation Frame Fired"), categories["scripting"]);
243 eventStyles[recordTypes.WebSocketCreate] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Create WebSocket"), categories["scripting"]);
244 eventStyles[recordTypes.WebSocketSendHandshakeRequest] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Send WebSocket Handshake"), categories["scripting"]);
245 eventStyles[recordTypes.WebSocketReceiveHandshakeResponse] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Receive WebSocket Handshake"), categories["scripting"]);
246 eventStyles[recordTypes.WebSocketDestroy] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Destroy WebSocket"), categories["scripting"]);
247 eventStyles[recordTypes.EmbedderCallback] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Embedder Callback"), categories["scripting"]);
248 eventStyles[recordTypes.DecodeImage] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Image Decode"), categories["painting"]);
249 eventStyles[recordTypes.ResizeImage] = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Image Resize"), categories["painting"]);
250 WebInspector.TracingTimelineUIUtils._eventStylesMap = eventStyles;
254 WebInspector.TracingTimelineUIUtils._coalescableRecordTypes = {};
255 WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[WebInspector.TracingTimelineModel.RecordType.Layout] = 1;
256 WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[WebInspector.TracingTimelineModel.RecordType.Paint] = 1;
257 WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[WebInspector.TracingTimelineModel.RecordType.RasterTask] = 1;
258 WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[WebInspector.TracingTimelineModel.RecordType.DecodeImage] = 1;
259 WebInspector.TracingTimelineUIUtils._coalescableRecordTypes[WebInspector.TracingTimelineModel.RecordType.ResizeImage] = 1;
262 * @param {!WebInspector.TracingModel.Event} event
263 * @return {!{title: string, category: !WebInspector.TimelineCategory}}
265 WebInspector.TracingTimelineUIUtils.eventStyle = function(event)
267 var eventStyles = WebInspector.TracingTimelineUIUtils._initEventStyles();
268 if (event.category === WebInspector.TracingModel.ConsoleEventCategory)
269 return { title: event.name, category: WebInspector.TimelineUIUtils.categories()["scripting"] };
271 var result = eventStyles[event.name];
273 result = new WebInspector.TimelineRecordStyle(WebInspector.UIString("Unknown: %s", event.name), WebInspector.TimelineUIUtils.categories()["other"]);
274 eventStyles[event.name] = result;
280 * @param {!WebInspector.TracingModel.Event} event
283 WebInspector.TracingTimelineUIUtils.markerEventColor = function(event)
285 var red = "rgb(255, 0, 0)";
286 var blue = "rgb(0, 0, 255)";
287 var orange = "rgb(255, 178, 23)";
288 var green = "rgb(0, 130, 0)";
290 if (event.category === WebInspector.TracingModel.ConsoleEventCategory)
293 var recordTypes = WebInspector.TracingTimelineModel.RecordType;
294 var eventName = event.name;
296 case recordTypes.MarkDOMContent: return blue;
297 case recordTypes.MarkLoad: return red;
298 case recordTypes.MarkFirstPaint: return green;
299 case recordTypes.TimeStamp: return orange;
305 * @param {!WebInspector.TracingModel.Event} event
306 * @param {!WebInspector.TimelineModel} model
309 WebInspector.TracingTimelineUIUtils.eventTitle = function(event, model)
311 var title = WebInspector.TracingTimelineUIUtils.eventStyle(event).title;
312 if (event.category === WebInspector.TracingModel.ConsoleEventCategory)
314 if (event.name === WebInspector.TracingTimelineModel.RecordType.TimeStamp)
315 return WebInspector.UIString("%s: %s", title, event.args["data"]["message"]);
316 if (WebInspector.TracingTimelineUIUtils.isMarkerEvent(event)) {
317 var startTime = Number.millisToString(event.startTime - model.minimumRecordTime());
318 return WebInspector.UIString("%s at %s", title, startTime);
324 * @param {!WebInspector.TracingModel.Event} event
327 WebInspector.TracingTimelineUIUtils.isMarkerEvent = function(event)
329 var recordTypes = WebInspector.TracingTimelineModel.RecordType;
330 switch (event.name) {
331 case recordTypes.TimeStamp:
332 case recordTypes.MarkFirstPaint:
334 case recordTypes.MarkDOMContent:
335 case recordTypes.MarkLoad:
336 return event.args["data"]["isMainFrame"];
343 * @param {!WebInspector.TracingModel.Event} event
344 * @param {!WebInspector.Linkifier} linkifier
347 WebInspector.TracingTimelineUIUtils.buildDetailsNodeForTraceEvent = function(event, linkifier)
349 var recordType = WebInspector.TracingTimelineModel.RecordType;
350 var target = event.thread.target();
353 var eventData = event.args["data"];
354 switch (event.name) {
355 case recordType.GCEvent:
356 var delta = event.args["usedHeapSizeBefore"] - event.args["usedHeapSizeAfter"];
357 detailsText = WebInspector.UIString("%s collected", Number.bytesToString(delta));
359 case recordType.TimerFire:
360 detailsText = eventData["timerId"];
362 case recordType.FunctionCall:
363 details = linkifyLocation(eventData["scriptId"], eventData["scriptName"], eventData["scriptLine"], 0);
365 case recordType.JSFrame:
366 details = linkifyLocation(eventData["scriptId"], eventData["url"], eventData["lineNumber"], eventData["columnNumber"]);
367 detailsText = WebInspector.CPUProfileDataModel.beautifyFunctionName(eventData["functionName"]);
369 case recordType.FireAnimationFrame:
370 detailsText = eventData["id"];
372 case recordType.EventDispatch:
373 detailsText = eventData ? eventData["type"] : null;
375 case recordType.Paint:
376 var width = WebInspector.TimelineUIUtils.quadWidth(eventData.clip);
377 var height = WebInspector.TimelineUIUtils.quadHeight(eventData.clip);
379 detailsText = WebInspector.UIString("%d\u2009\u00d7\u2009%d", width, height);
381 case recordType.TimerInstall:
382 case recordType.TimerRemove:
383 details = linkifyTopCallFrame();
384 detailsText = eventData["timerId"];
386 case recordType.RequestAnimationFrame:
387 case recordType.CancelAnimationFrame:
388 details = linkifyTopCallFrame();
389 detailsText = eventData["id"];
391 case recordType.ParseHTML:
392 case recordType.RecalculateStyles:
393 details = linkifyTopCallFrame();
395 case recordType.EvaluateScript:
396 var url = eventData["url"];
398 details = linkifyLocation("", url, eventData["lineNumber"], 0);
400 case recordType.XHRReadyStateChange:
401 case recordType.XHRLoad:
402 case recordType.ResourceSendRequest:
403 var url = eventData["url"];
405 detailsText = WebInspector.displayNameForURL(url);
407 case recordType.ResourceReceivedData:
408 case recordType.ResourceReceiveResponse:
409 case recordType.ResourceFinish:
410 var initiator = event.initiator;
412 var url = initiator.args["data"]["url"];
414 detailsText = WebInspector.displayNameForURL(url);
417 case recordType.EmbedderCallback:
418 detailsText = eventData["callbackName"];
421 case recordType.PaintImage:
422 case recordType.DecodeImage:
423 case recordType.ResizeImage:
424 case recordType.DecodeLazyPixelRef:
425 var url = event.imageURL;
427 detailsText = WebInspector.displayNameForURL(url);
431 if (event.category === WebInspector.TracingModel.ConsoleEventCategory)
434 details = linkifyTopCallFrame();
440 details.textContent = detailsText;
442 details = document.createTextNode(detailsText);
447 * @param {string} scriptId
448 * @param {string} url
449 * @param {number} lineNumber
450 * @param {number=} columnNumber
452 function linkifyLocation(scriptId, url, lineNumber, columnNumber)
457 // FIXME(62725): stack trace line/column numbers are one-based.
458 return linkifier.linkifyScriptLocation(target, scriptId, url, lineNumber - 1, (columnNumber ||1) - 1, "timeline-details");
464 function linkifyTopCallFrame()
466 var stackTrace = event.stackTrace;
468 var initiator = event.initiator;
470 stackTrace = initiator.stackTrace;
472 if (!stackTrace || !stackTrace.length)
474 return linkifier.linkifyConsoleCallFrame(target, stackTrace[0], "timeline-details");
479 * @param {!WebInspector.TracingModel.Event} event
480 * @param {!WebInspector.TracingTimelineModel} model
481 * @param {!WebInspector.Linkifier} linkifier
482 * @param {function(!DocumentFragment)} callback
484 WebInspector.TracingTimelineUIUtils.buildTraceEventDetails = function(event, model, linkifier, callback)
486 var target = event.thread.target();
487 var relatedNode = null;
488 var barrier = new CallbackBarrier();
489 if (!event.previewElement && target) {
491 WebInspector.DOMPresentationUtils.buildImagePreviewContents(target, event.imageURL, false, barrier.createCallback(saveImage));
492 else if (event.picture)
493 WebInspector.TracingTimelineUIUtils.buildPicturePreviewContent(event, barrier.createCallback(saveImage));
495 if (event.backendNodeId && target)
496 target.domModel.pushNodesByBackendIdsToFrontend([event.backendNodeId], barrier.createCallback(setRelatedNode));
497 barrier.callWhenDone(callbackWrapper);
500 * @param {!Element=} element
502 function saveImage(element)
504 event.previewElement = element || null;
508 * @param {?Array.<!DOMAgent.NodeId>} nodeIds
510 function setRelatedNode(nodeIds)
513 relatedNode = target.domModel.nodeForId(nodeIds[0]);
516 function callbackWrapper()
518 callback(WebInspector.TracingTimelineUIUtils._buildTraceEventDetailsSynchronously(event, model, linkifier, relatedNode));
523 * @param {!WebInspector.TracingModel.Event} event
524 * @param {!WebInspector.TracingTimelineModel} model
525 * @param {!WebInspector.Linkifier} linkifier
526 * @param {?WebInspector.DOMNode} relatedNode
527 * @return {!DocumentFragment}
529 WebInspector.TracingTimelineUIUtils._buildTraceEventDetailsSynchronously = function(event, model, linkifier, relatedNode)
531 var fragment = document.createDocumentFragment();
533 var hasChildren = WebInspector.TracingTimelineUIUtils._aggregatedStatsForTraceEvent(stats, model, event);
534 var pieChart = hasChildren ?
535 WebInspector.TimelineUIUtils.generatePieChart(stats, WebInspector.TracingTimelineUIUtils.eventStyle(event).category, event.selfTime) :
536 WebInspector.TimelineUIUtils.generatePieChart(stats);
537 fragment.appendChild(pieChart);
539 var recordTypes = WebInspector.TracingTimelineModel.RecordType;
541 // The messages may vary per event.name;
542 var callSiteStackTraceLabel;
544 var relatedNodeLabel;
546 var contentHelper = new WebInspector.TimelineDetailsContentHelper(event.thread.target(), linkifier, true);
547 contentHelper.appendTextRow(WebInspector.UIString("Self Time"), Number.millisToString(event.selfTime, true));
548 contentHelper.appendTextRow(WebInspector.UIString("Start Time"), Number.millisToString((event.startTime - model.minimumRecordTime())));
549 var eventData = event.args["data"];
550 var initiator = event.initiator;
552 switch (event.name) {
553 case recordTypes.GCEvent:
554 var delta = event.args["usedHeapSizeBefore"] - event.args["usedHeapSizeAfter"];
555 contentHelper.appendTextRow(WebInspector.UIString("Collected"), Number.bytesToString(delta));
557 case recordTypes.TimerFire:
558 callSiteStackTraceLabel = WebInspector.UIString("Timer installed");
559 // Fall-through intended.
561 case recordTypes.TimerInstall:
562 case recordTypes.TimerRemove:
563 contentHelper.appendTextRow(WebInspector.UIString("Timer ID"), eventData["timerId"]);
564 if (event.name === recordTypes.TimerInstall) {
565 contentHelper.appendTextRow(WebInspector.UIString("Timeout"), Number.millisToString(eventData["timeout"]));
566 contentHelper.appendTextRow(WebInspector.UIString("Repeats"), !eventData["singleShot"]);
569 case recordTypes.FireAnimationFrame:
570 callSiteStackTraceLabel = WebInspector.UIString("Animation frame requested");
571 contentHelper.appendTextRow(WebInspector.UIString("Callback ID"), eventData["id"]);
573 case recordTypes.FunctionCall:
574 if (eventData["scriptName"])
575 contentHelper.appendLocationRow(WebInspector.UIString("Location"), eventData["scriptName"], eventData["scriptLine"]);
577 case recordTypes.ResourceSendRequest:
578 case recordTypes.ResourceReceiveResponse:
579 case recordTypes.ResourceReceivedData:
580 case recordTypes.ResourceFinish:
581 var url = (event.name === recordTypes.ResourceSendRequest) ? eventData["url"] : initiator.args["data"]["url"];
583 contentHelper.appendElementRow(WebInspector.UIString("Resource"), WebInspector.linkifyResourceAsNode(url));
584 if (eventData["requestMethod"])
585 contentHelper.appendTextRow(WebInspector.UIString("Request Method"), eventData["requestMethod"]);
586 if (typeof eventData["statusCode"] === "number")
587 contentHelper.appendTextRow(WebInspector.UIString("Status Code"), eventData["statusCode"]);
588 if (eventData["mimeType"])
589 contentHelper.appendTextRow(WebInspector.UIString("MIME Type"), eventData["mimeType"]);
590 if (eventData["encodedDataLength"])
591 contentHelper.appendTextRow(WebInspector.UIString("Encoded Data Length"), WebInspector.UIString("%d Bytes", eventData["encodedDataLength"]));
593 case recordTypes.EvaluateScript:
594 var url = eventData["url"];
596 contentHelper.appendLocationRow(WebInspector.UIString("Script"), url, eventData["lineNumber"]);
598 case recordTypes.Paint:
599 var clip = eventData["clip"];
600 contentHelper.appendTextRow(WebInspector.UIString("Location"), WebInspector.UIString("(%d, %d)", clip[0], clip[1]));
601 var clipWidth = WebInspector.TimelineUIUtils.quadWidth(clip);
602 var clipHeight = WebInspector.TimelineUIUtils.quadHeight(clip);
603 contentHelper.appendTextRow(WebInspector.UIString("Dimensions"), WebInspector.UIString("%d × %d", clipWidth, clipHeight));
604 // Fall-through intended.
606 case recordTypes.PaintSetup:
607 case recordTypes.Rasterize:
608 case recordTypes.ScrollLayer:
609 relatedNodeLabel = WebInspector.UIString("Layer root");
611 case recordTypes.PaintImage:
612 case recordTypes.DecodeLazyPixelRef:
613 case recordTypes.DecodeImage:
614 case recordTypes.ResizeImage:
615 case recordTypes.DrawLazyPixelRef:
616 relatedNodeLabel = WebInspector.UIString("Image element");
618 contentHelper.appendElementRow(WebInspector.UIString("Image URL"), WebInspector.linkifyResourceAsNode(event.imageURL));
620 case recordTypes.RecalculateStyles: // We don't want to see default details.
621 contentHelper.appendTextRow(WebInspector.UIString("Elements affected"), event.args["elementCount"]);
622 callStackLabel = WebInspector.UIString("Styles recalculation forced");
624 case recordTypes.Layout:
625 var beginData = event.args["beginData"];
626 contentHelper.appendTextRow(WebInspector.UIString("Nodes that need layout"), beginData["dirtyObjects"]);
627 contentHelper.appendTextRow(WebInspector.UIString("Layout tree size"), beginData["totalObjects"]);
628 contentHelper.appendTextRow(WebInspector.UIString("Layout scope"),
629 beginData["partialLayout"] ? WebInspector.UIString("Partial") : WebInspector.UIString("Whole document"));
630 callSiteStackTraceLabel = WebInspector.UIString("Layout invalidated");
631 callStackLabel = WebInspector.UIString("Layout forced");
632 relatedNodeLabel = WebInspector.UIString("Layout root");
634 case recordTypes.ConsoleTime:
635 contentHelper.appendTextRow(WebInspector.UIString("Message"), event.name);
637 case recordTypes.WebSocketCreate:
638 case recordTypes.WebSocketSendHandshakeRequest:
639 case recordTypes.WebSocketReceiveHandshakeResponse:
640 case recordTypes.WebSocketDestroy:
641 var initiatorData = initiator ? initiator.args["data"] : eventData;
642 if (typeof initiatorData["webSocketURL"] !== "undefined")
643 contentHelper.appendTextRow(WebInspector.UIString("URL"), initiatorData["webSocketURL"]);
644 if (typeof initiatorData["webSocketProtocol"] !== "undefined")
645 contentHelper.appendTextRow(WebInspector.UIString("WebSocket Protocol"), initiatorData["webSocketProtocol"]);
646 if (typeof eventData["message"] !== "undefined")
647 contentHelper.appendTextRow(WebInspector.UIString("Message"), eventData["message"]);
649 case recordTypes.EmbedderCallback:
650 contentHelper.appendTextRow(WebInspector.UIString("Callback Function"), eventData["callbackName"]);
653 var detailsNode = WebInspector.TracingTimelineUIUtils.buildDetailsNodeForTraceEvent(event, linkifier);
655 contentHelper.appendElementRow(WebInspector.UIString("Details"), detailsNode);
660 contentHelper.appendElementRow(relatedNodeLabel || WebInspector.UIString("Related node"), WebInspector.DOMPresentationUtils.linkifyNodeReference(relatedNode));
662 if (eventData && eventData["scriptName"] && event.name !== recordTypes.FunctionCall)
663 contentHelper.appendLocationRow(WebInspector.UIString("Function Call"), eventData["scriptName"], eventData["scriptLine"]);
666 var callSiteStackTrace = initiator.stackTrace;
667 if (callSiteStackTrace)
668 contentHelper.appendStackTrace(callSiteStackTraceLabel || WebInspector.UIString("Call Site stack"), callSiteStackTrace);
670 var eventStackTrace = event.stackTrace;
672 contentHelper.appendStackTrace(callStackLabel || WebInspector.UIString("Call Stack"), eventStackTrace);
674 var warning = event.warning;
676 var div = document.createElement("div");
677 div.textContent = warning;
678 contentHelper.appendElementRow(WebInspector.UIString("Warning"), div);
680 if (event.previewElement)
681 contentHelper.appendElementRow(WebInspector.UIString("Preview"), event.previewElement);
682 fragment.appendChild(contentHelper.element);
687 * @param {!Object} total
688 * @param {!WebInspector.TracingTimelineModel} model
689 * @param {!WebInspector.TracingModel.Event} event
692 WebInspector.TracingTimelineUIUtils._aggregatedStatsForTraceEvent = function(total, model, event)
694 var events = model.inspectedTargetEvents();
696 * @param {number} startTime
697 * @param {!WebInspector.TracingModel.Event} e
700 function eventComparator(startTime, e)
702 return startTime - e.startTime;
704 var index = events.binaryIndexOf(event.startTime, eventComparator);
707 var hasChildren = false;
708 var endTime = event.endTime;
710 for (var i = index; i < events.length; i++) {
711 var nextEvent = events[i];
712 if (nextEvent.startTime >= endTime)
714 if (!nextEvent.selfTime)
716 if (nextEvent.thread !== event.thread)
720 var categoryName = WebInspector.TracingTimelineUIUtils.eventStyle(nextEvent).category.name;
721 total[categoryName] = (total[categoryName] || 0) + nextEvent.selfTime;
728 * @param {!WebInspector.TracingModel.Event} event
729 * @param {function(!Element=)} callback
731 WebInspector.TracingTimelineUIUtils.buildPicturePreviewContent = function(event, callback)
734 new WebInspector.LayerPaintEvent(event).loadPicture(onSnapshotLoaded);
736 * @param {?Array.<number>} rect
737 * @param {?WebInspector.PaintProfilerSnapshot} snapshot
739 function onSnapshotLoaded(rect, snapshot)
745 snapshot.requestImage(null, null, 1, onGotImage);
750 * @param {string=} imageURL
752 function onGotImage(imageURL)
758 var container = document.createElement("div");
759 container.className = "image-preview-container";
760 var img = container.createChild("img");
767 * @param {string} recordType
768 * @param {string=} title
771 WebInspector.TracingTimelineUIUtils._createEventDivider = function(recordType, title)
773 var eventDivider = document.createElement("div");
774 eventDivider.className = "resources-event-divider";
775 var recordTypes = WebInspector.TracingTimelineModel.RecordType;
777 if (recordType === recordTypes.MarkDOMContent)
778 eventDivider.className += " resources-blue-divider";
779 else if (recordType === recordTypes.MarkLoad)
780 eventDivider.className += " resources-red-divider";
781 else if (recordType === recordTypes.MarkFirstPaint)
782 eventDivider.className += " resources-green-divider";
783 else if (recordType === recordTypes.TimeStamp || recordType === recordTypes.ConsoleTime)
784 eventDivider.className += " resources-orange-divider";
785 else if (recordType === recordTypes.BeginFrame)
786 eventDivider.className += " timeline-frame-divider";
789 eventDivider.title = title;
795 * @return {!Array.<string>}
797 WebInspector.TracingTimelineUIUtils._visibleTypes = function()
799 var eventStyles = WebInspector.TracingTimelineUIUtils._initEventStyles();
801 for (var name in eventStyles) {
802 if (!eventStyles[name].hidden)
809 * @return {!WebInspector.TracingTimelineModel.Filter}
811 WebInspector.TracingTimelineUIUtils.hiddenEventsFilter = function()
813 return new WebInspector.TracingTimelineModel.InclusiveEventNameFilter(WebInspector.TracingTimelineUIUtils._visibleTypes());