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.TimelineUIUtilsImpl = function()
11 WebInspector.TimelineUIUtils.call(this);
14 WebInspector.TimelineUIUtilsImpl.prototype = {
16 * @param {!WebInspector.TimelineModel.Record} record
19 isBeginFrame: function(record)
21 return record.type() === WebInspector.TimelineModel.RecordType.BeginFrame;
24 * @param {!WebInspector.TimelineModel.Record} record
27 isProgram: function(record)
29 return record.type() === WebInspector.TimelineModel.RecordType.Program;
32 * @param {string} recordType
35 isCoalescable: function(recordType)
37 return !!WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes[recordType];
41 * @param {!WebInspector.TimelineModel.Record} record
44 isEventDivider: function(record)
46 return WebInspector.TimelineUIUtilsImpl.isEventDivider(record);
50 * @param {!WebInspector.TimelineModel.Record} record
53 countersForRecord: function(record)
55 return record.type() === WebInspector.TimelineModel.RecordType.UpdateCounters ? record.data() : null;
59 * @param {!WebInspector.TimelineModel.Record} record
62 highlightQuadForRecord: function(record)
64 var recordTypes = WebInspector.TimelineModel.RecordType;
65 switch(record.type()) {
66 case recordTypes.Layout:
67 return record.data().root;
68 case recordTypes.Paint:
69 return record.data().clip;
76 * @param {!WebInspector.TimelineModel.Record} record
79 titleForRecord: function(record)
81 return WebInspector.TimelineUIUtilsImpl._recordTitle(record);
85 * @param {!WebInspector.TimelineModel.Record} record
86 * @return {!WebInspector.TimelineCategory}
88 categoryForRecord: function(record)
90 return WebInspector.TimelineUIUtilsImpl.recordStyle(record).category;
94 * @param {!WebInspector.TimelineModel.Record} record
95 * @param {!WebInspector.Linkifier} linkifier
98 buildDetailsNode: function(record, linkifier)
100 return WebInspector.TimelineUIUtilsImpl.buildDetailsNode(record, linkifier);
104 * @param {!WebInspector.TimelineModel.Record} record
105 * @param {!WebInspector.TimelineModel} model
106 * @param {!WebInspector.Linkifier} linkifier
107 * @param {function(!DocumentFragment)} callback
109 generateDetailsContent: function(record, model, linkifier, callback)
111 WebInspector.TimelineUIUtilsImpl.generateDetailsContent(record, model, linkifier, callback);
117 createBeginFrameDivider: function()
119 return this.createEventDivider(WebInspector.TimelineModel.RecordType.BeginFrame);
123 * @param {string} recordType
124 * @param {string=} title
127 createEventDivider: function(recordType, title)
129 return WebInspector.TimelineUIUtilsImpl._createEventDivider(recordType, title);
133 * @param {!WebInspector.TimelineModel.Record} record
134 * @param {!RegExp} regExp
137 testContentMatching: function(record, regExp)
139 var tokens = [WebInspector.TimelineUIUtilsImpl._recordTitle(record)];
140 var data = record.data();
141 for (var key in data)
142 tokens.push(data[key])
143 return regExp.test(tokens.join("|"));
147 * @param {!Object} total
148 * @param {!WebInspector.TimelineModel.Record} record
150 aggregateTimeForRecord: function(total, record)
152 WebInspector.TimelineUIUtilsImpl.aggregateTimeForRecord(total, record);
156 * @return {!WebInspector.TimelineModel.Filter}
158 hiddenRecordsFilter: function()
160 var recordTypes = WebInspector.TimelineModel.RecordType;
161 var hiddenRecords = [
162 recordTypes.ActivateLayerTree,
163 recordTypes.BeginFrame,
164 recordTypes.DrawFrame,
166 recordTypes.InvalidateLayout,
167 recordTypes.MarkDOMContent,
168 recordTypes.MarkFirstPaint,
169 recordTypes.MarkLoad,
170 recordTypes.RequestMainThreadFrame,
171 recordTypes.ScheduleStyleRecalculation,
172 recordTypes.UpdateCounters
174 return new WebInspector.TimelineRecordHiddenTypeFilter(hiddenRecords);
177 __proto__: WebInspector.TimelineUIUtils.prototype
181 WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes = {};
182 WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.Layout] = 1;
183 WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.Paint] = 1;
184 WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.Rasterize] = 1;
185 WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.DecodeImage] = 1;
186 WebInspector.TimelineUIUtilsImpl._coalescableRecordTypes[WebInspector.TimelineModel.RecordType.ResizeImage] = 1;
189 * @return {!Object.<string, !{title: string, category: !WebInspector.TimelineCategory}>}
191 WebInspector.TimelineUIUtils._initRecordStyles = function()
193 if (WebInspector.TimelineUIUtils._recordStylesMap)
194 return WebInspector.TimelineUIUtils._recordStylesMap;
196 var recordTypes = WebInspector.TimelineModel.RecordType;
197 var categories = WebInspector.TimelineUIUtils.categories();
199 var recordStyles = {};
200 recordStyles[recordTypes.Root] = { title: "#root", category: categories["loading"] };
201 recordStyles[recordTypes.Program] = { title: WebInspector.UIString("Other"), category: categories["other"] };
202 recordStyles[recordTypes.EventDispatch] = { title: WebInspector.UIString("Event"), category: categories["scripting"] };
203 recordStyles[recordTypes.BeginFrame] = { title: WebInspector.UIString("Frame Start"), category: categories["rendering"] };
204 recordStyles[recordTypes.ScheduleStyleRecalculation] = { title: WebInspector.UIString("Schedule Style Recalculation"), category: categories["rendering"] };
205 recordStyles[recordTypes.RecalculateStyles] = { title: WebInspector.UIString("Recalculate Style"), category: categories["rendering"] };
206 recordStyles[recordTypes.InvalidateLayout] = { title: WebInspector.UIString("Invalidate Layout"), category: categories["rendering"] };
207 recordStyles[recordTypes.Layout] = { title: WebInspector.UIString("Layout"), category: categories["rendering"] };
208 recordStyles[recordTypes.UpdateLayerTree] = { title: WebInspector.UIString("Update Layer Tree"), category: categories["rendering"] };
209 recordStyles[recordTypes.PaintSetup] = { title: WebInspector.UIString("Paint Setup"), category: categories["painting"] };
210 recordStyles[recordTypes.Paint] = { title: WebInspector.UIString("Paint"), category: categories["painting"] };
211 recordStyles[recordTypes.Rasterize] = { title: WebInspector.UIString("Paint"), category: categories["painting"] };
212 recordStyles[recordTypes.ScrollLayer] = { title: WebInspector.UIString("Scroll"), category: categories["rendering"] };
213 recordStyles[recordTypes.DecodeImage] = { title: WebInspector.UIString("Image Decode"), category: categories["painting"] };
214 recordStyles[recordTypes.ResizeImage] = { title: WebInspector.UIString("Image Resize"), category: categories["painting"] };
215 recordStyles[recordTypes.CompositeLayers] = { title: WebInspector.UIString("Composite Layers"), category: categories["painting"] };
216 recordStyles[recordTypes.ParseHTML] = { title: WebInspector.UIString("Parse HTML"), category: categories["loading"] };
217 recordStyles[recordTypes.TimerInstall] = { title: WebInspector.UIString("Install Timer"), category: categories["scripting"] };
218 recordStyles[recordTypes.TimerRemove] = { title: WebInspector.UIString("Remove Timer"), category: categories["scripting"] };
219 recordStyles[recordTypes.TimerFire] = { title: WebInspector.UIString("Timer Fired"), category: categories["scripting"] };
220 recordStyles[recordTypes.XHRReadyStateChange] = { title: WebInspector.UIString("XHR Ready State Change"), category: categories["scripting"] };
221 recordStyles[recordTypes.XHRLoad] = { title: WebInspector.UIString("XHR Load"), category: categories["scripting"] };
222 recordStyles[recordTypes.EvaluateScript] = { title: WebInspector.UIString("Evaluate Script"), category: categories["scripting"] };
223 recordStyles[recordTypes.ResourceSendRequest] = { title: WebInspector.UIString("Send Request"), category: categories["loading"] };
224 recordStyles[recordTypes.ResourceReceiveResponse] = { title: WebInspector.UIString("Receive Response"), category: categories["loading"] };
225 recordStyles[recordTypes.ResourceFinish] = { title: WebInspector.UIString("Finish Loading"), category: categories["loading"] };
226 recordStyles[recordTypes.FunctionCall] = { title: WebInspector.UIString("Function Call"), category: categories["scripting"] };
227 recordStyles[recordTypes.ResourceReceivedData] = { title: WebInspector.UIString("Receive Data"), category: categories["loading"] };
228 recordStyles[recordTypes.GCEvent] = { title: WebInspector.UIString("GC Event"), category: categories["scripting"] };
229 recordStyles[recordTypes.JSFrame] = { title: WebInspector.UIString("JS Frame"), category: categories["scripting"] };
230 recordStyles[recordTypes.MarkDOMContent] = { title: WebInspector.UIString("DOMContentLoaded event"), category: categories["scripting"] };
231 recordStyles[recordTypes.MarkLoad] = { title: WebInspector.UIString("Load event"), category: categories["scripting"] };
232 recordStyles[recordTypes.MarkFirstPaint] = { title: WebInspector.UIString("First paint"), category: categories["painting"] };
233 recordStyles[recordTypes.TimeStamp] = { title: WebInspector.UIString("Stamp"), category: categories["scripting"] };
234 recordStyles[recordTypes.ConsoleTime] = { title: WebInspector.UIString("Console Time"), category: categories["scripting"] };
235 recordStyles[recordTypes.RequestAnimationFrame] = { title: WebInspector.UIString("Request Animation Frame"), category: categories["scripting"] };
236 recordStyles[recordTypes.CancelAnimationFrame] = { title: WebInspector.UIString("Cancel Animation Frame"), category: categories["scripting"] };
237 recordStyles[recordTypes.FireAnimationFrame] = { title: WebInspector.UIString("Animation Frame Fired"), category: categories["scripting"] };
238 recordStyles[recordTypes.WebSocketCreate] = { title: WebInspector.UIString("Create WebSocket"), category: categories["scripting"] };
239 recordStyles[recordTypes.WebSocketSendHandshakeRequest] = { title: WebInspector.UIString("Send WebSocket Handshake"), category: categories["scripting"] };
240 recordStyles[recordTypes.WebSocketReceiveHandshakeResponse] = { title: WebInspector.UIString("Receive WebSocket Handshake"), category: categories["scripting"] };
241 recordStyles[recordTypes.WebSocketDestroy] = { title: WebInspector.UIString("Destroy WebSocket"), category: categories["scripting"] };
242 recordStyles[recordTypes.EmbedderCallback] = { title: WebInspector.UIString("Embedder Callback"), category: categories["scripting"] };
244 WebInspector.TimelineUIUtils._recordStylesMap = recordStyles;
249 * @param {!WebInspector.TimelineModel.Record} record
250 * @return {!{title: string, category: !WebInspector.TimelineCategory}}
252 WebInspector.TimelineUIUtilsImpl.recordStyle = function(record)
254 var type = record.type();
255 var recordStyles = WebInspector.TimelineUIUtils._initRecordStyles();
256 var result = recordStyles[type];
259 title: WebInspector.UIString("Unknown: %s", type),
260 category: WebInspector.TimelineUIUtils.categories()["other"]
262 recordStyles[type] = result;
268 * @param {!WebInspector.TimelineModel.Record} record
271 WebInspector.TimelineUIUtilsImpl._recordTitle = function(record)
273 var recordData = record.data();
274 if (record.type() === WebInspector.TimelineModel.RecordType.TimeStamp)
275 return recordData["message"];
276 if (record.type() === WebInspector.TimelineModel.RecordType.JSFrame)
277 return recordData["functionName"];
278 var title = WebInspector.TimelineUIUtilsImpl.recordStyle(record).title;
279 if (WebInspector.TimelineUIUtilsImpl.isEventDivider(record)) {
280 var startTime = Number.millisToString(record.startTime() - record._model.minimumRecordTime());
281 return WebInspector.UIString("%s at %s", title, startTime);
287 * @param {!WebInspector.TimelineModel.Record} record
290 WebInspector.TimelineUIUtilsImpl.isEventDivider = function(record)
292 var recordTypes = WebInspector.TimelineModel.RecordType;
293 if (record.type() === recordTypes.TimeStamp)
295 if (record.type() === recordTypes.MarkFirstPaint)
297 if (record.type() === recordTypes.MarkDOMContent || record.type() === recordTypes.MarkLoad)
298 return record.data()["isMainFrame"];
303 * @param {!Object} total
304 * @param {!WebInspector.TimelineModel.Record} record
306 WebInspector.TimelineUIUtilsImpl.aggregateTimeForRecord = function(total, record)
308 var children = record.children();
309 for (var i = 0; i < children.length; ++i)
310 WebInspector.TimelineUIUtilsImpl.aggregateTimeForRecord(total, children[i]);
311 var categoryName = WebInspector.TimelineUIUtilsImpl.recordStyle(record).category.name;
312 total[categoryName] = (total[categoryName] || 0) + record.selfTime();
316 * @param {!WebInspector.TimelineModel.Record} record
317 * @param {!WebInspector.Linkifier} linkifier
320 WebInspector.TimelineUIUtilsImpl.buildDetailsNode = function(record, linkifier)
324 var recordData = record.data();
325 switch (record.type()) {
326 case WebInspector.TimelineModel.RecordType.GCEvent:
327 detailsText = WebInspector.UIString("%s collected", Number.bytesToString(recordData["usedHeapSizeDelta"]));
329 case WebInspector.TimelineModel.RecordType.TimerFire:
330 detailsText = recordData["timerId"];
332 case WebInspector.TimelineModel.RecordType.FunctionCall:
333 details = linkifyLocation(recordData["scriptId"], recordData["scriptName"], recordData["scriptLine"], 0);
335 case WebInspector.TimelineModel.RecordType.FireAnimationFrame:
336 detailsText = recordData["id"];
338 case WebInspector.TimelineModel.RecordType.EventDispatch:
339 detailsText = recordData ? recordData["type"] : null;
341 case WebInspector.TimelineModel.RecordType.Paint:
342 var width = WebInspector.TimelineUIUtils.quadWidth(recordData.clip);
343 var height = WebInspector.TimelineUIUtils.quadHeight(recordData.clip);
345 detailsText = WebInspector.UIString("%d\u2009\u00d7\u2009%d", width, height);
347 case WebInspector.TimelineModel.RecordType.TimerInstall:
348 case WebInspector.TimelineModel.RecordType.TimerRemove:
349 details = linkifyTopCallFrame();
350 detailsText = recordData["timerId"];
352 case WebInspector.TimelineModel.RecordType.RequestAnimationFrame:
353 case WebInspector.TimelineModel.RecordType.CancelAnimationFrame:
354 details = linkifyTopCallFrame();
355 detailsText = recordData["id"];
357 case WebInspector.TimelineModel.RecordType.ParseHTML:
358 case WebInspector.TimelineModel.RecordType.RecalculateStyles:
359 details = linkifyTopCallFrame();
361 case WebInspector.TimelineModel.RecordType.EvaluateScript:
362 var url = recordData["url"];
364 details = linkifyLocation("", url, recordData["lineNumber"], 0);
366 case WebInspector.TimelineModel.RecordType.XHRReadyStateChange:
367 case WebInspector.TimelineModel.RecordType.XHRLoad:
368 case WebInspector.TimelineModel.RecordType.ResourceSendRequest:
369 case WebInspector.TimelineModel.RecordType.DecodeImage:
370 case WebInspector.TimelineModel.RecordType.ResizeImage:
371 var url = recordData["url"];
373 detailsText = WebInspector.displayNameForURL(url);
375 case WebInspector.TimelineModel.RecordType.ResourceReceivedData:
376 case WebInspector.TimelineModel.RecordType.ResourceReceiveResponse:
377 case WebInspector.TimelineModel.RecordType.ResourceFinish:
378 var initiator = record.initiator();
380 var url = initiator.data()["url"];
382 detailsText = WebInspector.displayNameForURL(url);
385 case WebInspector.TimelineModel.RecordType.ConsoleTime:
386 detailsText = recordData["message"];
388 case WebInspector.TimelineModel.RecordType.EmbedderCallback:
389 detailsText = recordData["callbackName"];
392 details = linkifyTopCallFrame();
396 if (!details && detailsText)
397 details = document.createTextNode(detailsText);
401 * @param {string} scriptId
402 * @param {string} url
403 * @param {number} lineNumber
404 * @param {number=} columnNumber
406 function linkifyLocation(scriptId, url, lineNumber, columnNumber)
411 // FIXME(62725): stack trace line/column numbers are one-based.
412 columnNumber = columnNumber ? columnNumber - 1 : 0;
413 return linkifier.linkifyScriptLocation(record.target(), scriptId, url, lineNumber - 1, columnNumber, "timeline-details");
419 function linkifyTopCallFrame()
421 if (record.stackTrace())
422 return linkifier.linkifyConsoleCallFrame(record.target(), record.stackTrace()[0], "timeline-details");
423 if (record.callSiteStackTrace())
424 return linkifier.linkifyConsoleCallFrame(record.target(), record.callSiteStackTrace()[0], "timeline-details");
430 * @param {string=} recordType
433 WebInspector.TimelineUIUtilsImpl._needsPreviewElement = function(recordType)
437 const recordTypes = WebInspector.TimelineModel.RecordType;
438 switch (recordType) {
439 case recordTypes.ResourceSendRequest:
440 case recordTypes.ResourceReceiveResponse:
441 case recordTypes.ResourceReceivedData:
442 case recordTypes.ResourceFinish:
450 * @param {!WebInspector.TimelineModel.Record} record
451 * @param {!WebInspector.TimelineModel} model
452 * @param {!WebInspector.Linkifier} linkifier
453 * @param {function(!DocumentFragment)} callback
455 WebInspector.TimelineUIUtilsImpl.generateDetailsContent = function(record, model, linkifier, callback)
457 var imageElement = /** @type {?Element} */ (record.getUserObject("TimelineUIUtils::preview-element") || null);
458 var relatedNode = null;
459 var recordData = record.data();
460 var barrier = new CallbackBarrier();
461 var target = record.target();
462 if (!imageElement && WebInspector.TimelineUIUtilsImpl._needsPreviewElement(record.type()) && target)
463 WebInspector.DOMPresentationUtils.buildImagePreviewContents(target, recordData["url"], false, barrier.createCallback(saveImage));
464 if (recordData["backendNodeId"] && target)
465 target.domModel.pushNodesByBackendIdsToFrontend([recordData["backendNodeId"]], barrier.createCallback(setRelatedNode));
466 barrier.callWhenDone(callbackWrapper);
469 * @param {!Element=} element
471 function saveImage(element)
473 imageElement = element || null;
474 record.setUserObject("TimelineUIUtils::preview-element", element);
478 * @param {?Array.<!DOMAgent.NodeId>} nodeIds
480 function setRelatedNode(nodeIds)
482 if (nodeIds && target)
483 relatedNode = target.domModel.nodeForId(nodeIds[0]);
486 function callbackWrapper()
488 callback(WebInspector.TimelineUIUtilsImpl._generateDetailsContentSynchronously(record, model, linkifier, imageElement, relatedNode));
493 * @param {!WebInspector.TimelineModel.Record} record
494 * @param {!WebInspector.TimelineModel} model
495 * @param {!WebInspector.Linkifier} linkifier
496 * @param {?Element} imagePreviewElement
497 * @param {?WebInspector.DOMNode} relatedNode
498 * @return {!DocumentFragment}
500 WebInspector.TimelineUIUtilsImpl._generateDetailsContentSynchronously = function(record, model, linkifier, imagePreviewElement, relatedNode)
502 var fragment = document.createDocumentFragment();
503 var aggregatedStats = {};
504 WebInspector.TimelineUIUtilsImpl.aggregateTimeForRecord(aggregatedStats, record);
505 if (record.children().length)
506 fragment.appendChild(WebInspector.TimelineUIUtils.generatePieChart(aggregatedStats, WebInspector.TimelineUIUtilsImpl.recordStyle(record).category, record.selfTime()));
508 fragment.appendChild(WebInspector.TimelineUIUtils.generatePieChart(aggregatedStats));
510 const recordTypes = WebInspector.TimelineModel.RecordType;
512 // The messages may vary per record.type();
513 var callSiteStackTraceLabel;
515 var relatedNodeLabel;
517 var contentHelper = new WebInspector.TimelineDetailsContentHelper(record.target(), linkifier, true);
518 contentHelper.appendTextRow(WebInspector.UIString("Self Time"), Number.millisToString(record.selfTime(), true));
519 contentHelper.appendTextRow(WebInspector.UIString("Start Time"), Number.millisToString(record.startTime() - model.minimumRecordTime()));
520 var recordData = record.data();
522 switch (record.type()) {
523 case recordTypes.GCEvent:
524 contentHelper.appendTextRow(WebInspector.UIString("Collected"), Number.bytesToString(recordData["usedHeapSizeDelta"]));
526 case recordTypes.TimerFire:
527 callSiteStackTraceLabel = WebInspector.UIString("Timer installed");
528 // Fall-through intended.
530 case recordTypes.TimerInstall:
531 case recordTypes.TimerRemove:
532 contentHelper.appendTextRow(WebInspector.UIString("Timer ID"), recordData["timerId"]);
533 if (record.type() === recordTypes.TimerInstall) {
534 contentHelper.appendTextRow(WebInspector.UIString("Timeout"), Number.millisToString(recordData["timeout"]));
535 contentHelper.appendTextRow(WebInspector.UIString("Repeats"), !recordData["singleShot"]);
538 case recordTypes.FireAnimationFrame:
539 callSiteStackTraceLabel = WebInspector.UIString("Animation frame requested");
540 contentHelper.appendTextRow(WebInspector.UIString("Callback ID"), recordData["id"]);
542 case recordTypes.FunctionCall:
543 if (recordData["scriptName"])
544 contentHelper.appendLocationRow(WebInspector.UIString("Location"), recordData["scriptName"], recordData["scriptLine"]);
546 case recordTypes.ResourceSendRequest:
547 case recordTypes.ResourceReceiveResponse:
548 case recordTypes.ResourceReceivedData:
549 case recordTypes.ResourceFinish:
551 if (record.type() === recordTypes.ResourceSendRequest)
552 url = recordData["url"];
553 else if (record.initiator())
554 url = record.initiator().data()["url"];
556 contentHelper.appendElementRow(WebInspector.UIString("Resource"), WebInspector.linkifyResourceAsNode(url));
557 if (imagePreviewElement)
558 contentHelper.appendElementRow(WebInspector.UIString("Preview"), imagePreviewElement);
559 if (recordData["requestMethod"])
560 contentHelper.appendTextRow(WebInspector.UIString("Request Method"), recordData["requestMethod"]);
561 if (typeof recordData["statusCode"] === "number")
562 contentHelper.appendTextRow(WebInspector.UIString("Status Code"), recordData["statusCode"]);
563 if (recordData["mimeType"])
564 contentHelper.appendTextRow(WebInspector.UIString("MIME Type"), recordData["mimeType"]);
565 if (recordData["encodedDataLength"])
566 contentHelper.appendTextRow(WebInspector.UIString("Encoded Data Length"), WebInspector.UIString("%d Bytes", recordData["encodedDataLength"]));
568 case recordTypes.EvaluateScript:
569 var url = recordData["url"];
571 contentHelper.appendLocationRow(WebInspector.UIString("Script"), url, recordData["lineNumber"]);
573 case recordTypes.Paint:
574 var clip = recordData["clip"];
575 contentHelper.appendTextRow(WebInspector.UIString("Location"), WebInspector.UIString("(%d, %d)", clip[0], clip[1]));
576 var clipWidth = WebInspector.TimelineUIUtils.quadWidth(clip);
577 var clipHeight = WebInspector.TimelineUIUtils.quadHeight(clip);
578 contentHelper.appendTextRow(WebInspector.UIString("Dimensions"), WebInspector.UIString("%d × %d", clipWidth, clipHeight));
579 // Fall-through intended.
581 case recordTypes.PaintSetup:
582 case recordTypes.Rasterize:
583 case recordTypes.ScrollLayer:
584 relatedNodeLabel = WebInspector.UIString("Layer root");
586 case recordTypes.DecodeImage:
587 case recordTypes.ResizeImage:
588 relatedNodeLabel = WebInspector.UIString("Image element");
589 var url = recordData["url"];
591 contentHelper.appendElementRow(WebInspector.UIString("Image URL"), WebInspector.linkifyResourceAsNode(url));
593 case recordTypes.RecalculateStyles: // We don't want to see default details.
594 if (recordData["elementCount"])
595 contentHelper.appendTextRow(WebInspector.UIString("Elements affected"), recordData["elementCount"]);
596 callStackLabel = WebInspector.UIString("Styles recalculation forced");
598 case recordTypes.Layout:
599 if (recordData["dirtyObjects"])
600 contentHelper.appendTextRow(WebInspector.UIString("Nodes that need layout"), recordData["dirtyObjects"]);
601 if (recordData["totalObjects"])
602 contentHelper.appendTextRow(WebInspector.UIString("Layout tree size"), recordData["totalObjects"]);
603 if (typeof recordData["partialLayout"] === "boolean") {
604 contentHelper.appendTextRow(WebInspector.UIString("Layout scope"),
605 recordData["partialLayout"] ? WebInspector.UIString("Partial") : WebInspector.UIString("Whole document"));
607 callSiteStackTraceLabel = WebInspector.UIString("Layout invalidated");
608 callStackLabel = WebInspector.UIString("Layout forced");
609 relatedNodeLabel = WebInspector.UIString("Layout root");
611 case recordTypes.ConsoleTime:
612 contentHelper.appendTextRow(WebInspector.UIString("Message"), recordData["message"]);
614 case recordTypes.WebSocketCreate:
615 case recordTypes.WebSocketSendHandshakeRequest:
616 case recordTypes.WebSocketReceiveHandshakeResponse:
617 case recordTypes.WebSocketDestroy:
618 var initiatorData = record.initiator() ? record.initiator().data() : recordData;
619 if (typeof initiatorData["webSocketURL"] !== "undefined")
620 contentHelper.appendTextRow(WebInspector.UIString("URL"), initiatorData["webSocketURL"]);
621 if (typeof initiatorData["webSocketProtocol"] !== "undefined")
622 contentHelper.appendTextRow(WebInspector.UIString("WebSocket Protocol"), initiatorData["webSocketProtocol"]);
623 if (typeof recordData["message"] !== "undefined")
624 contentHelper.appendTextRow(WebInspector.UIString("Message"), recordData["message"]);
626 case recordTypes.EmbedderCallback:
627 contentHelper.appendTextRow(WebInspector.UIString("Callback Function"), recordData["callbackName"]);
630 var detailsNode = WebInspector.TimelineUIUtilsImpl.buildDetailsNode(record, linkifier);
632 contentHelper.appendElementRow(WebInspector.UIString("Details"), detailsNode);
637 contentHelper.appendElementRow(relatedNodeLabel || WebInspector.UIString("Related node"), WebInspector.DOMPresentationUtils.linkifyNodeReference(relatedNode));
639 if (recordData["scriptName"] && record.type() !== recordTypes.FunctionCall)
640 contentHelper.appendLocationRow(WebInspector.UIString("Function Call"), recordData["scriptName"], recordData["scriptLine"]);
641 var callSiteStackTrace = record.callSiteStackTrace();
642 if (callSiteStackTrace)
643 contentHelper.appendStackTrace(callSiteStackTraceLabel || WebInspector.UIString("Call Site stack"), callSiteStackTrace);
644 var recordStackTrace = record.stackTrace();
645 if (recordStackTrace)
646 contentHelper.appendStackTrace(callStackLabel || WebInspector.UIString("Call Stack"), recordStackTrace);
648 if (record.warnings()) {
649 var ul = document.createElement("ul");
650 for (var i = 0; i < record.warnings().length; ++i)
651 ul.createChild("li").textContent = record.warnings()[i];
652 contentHelper.appendElementRow(WebInspector.UIString("Warning"), ul);
654 fragment.appendChild(contentHelper.element);
659 * @param {string} recordType
660 * @param {string=} title
663 WebInspector.TimelineUIUtilsImpl._createEventDivider = function(recordType, title)
665 var eventDivider = document.createElement("div");
666 eventDivider.className = "resources-event-divider";
667 var recordTypes = WebInspector.TimelineModel.RecordType;
669 if (recordType === recordTypes.MarkDOMContent)
670 eventDivider.className += " resources-blue-divider";
671 else if (recordType === recordTypes.MarkLoad)
672 eventDivider.className += " resources-red-divider";
673 else if (recordType === recordTypes.MarkFirstPaint)
674 eventDivider.className += " resources-green-divider";
675 else if (recordType === recordTypes.TimeStamp)
676 eventDivider.className += " resources-orange-divider";
677 else if (recordType === recordTypes.BeginFrame)
678 eventDivider.className += " timeline-frame-divider";
681 eventDivider.title = title;