blob: de0d78f1b35278decc6b2b4032dd73f09444eb66 [file] [log] [blame]
Blink Reformat4c46d092018-04-07 15:32:371/*
2 * Copyright (C) 2011 Google Inc. All rights reserved.
3 * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
4 * Copyright (C) 2009 Joseph Pecoraro
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 *
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
16 * its contributors may be used to endorse or promote products derived
17 * from this software without specific prior written permission.
18 *
19 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
20 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
21 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
22 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
23 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
24 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
25 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
26 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
28 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 */
30/**
31 * @implements {Console.ConsoleViewportElement}
32 * @unrestricted
33 */
34Console.ConsoleViewMessage = class {
35 /**
36 * @param {!SDK.ConsoleMessage} consoleMessage
37 * @param {!Components.Linkifier} linkifier
38 * @param {!ProductRegistry.BadgePool} badgePool
39 * @param {number} nestingLevel
40 */
41 constructor(consoleMessage, linkifier, badgePool, nestingLevel) {
42 this._message = consoleMessage;
43 this._linkifier = linkifier;
44 this._badgePool = badgePool;
45 this._repeatCount = 1;
46 this._closeGroupDecorationCount = 0;
47 this._nestingLevel = nestingLevel;
48
49 /** @type {?DataGrid.DataGrid} */
50 this._dataGrid = null;
51 this._previewFormatter = new ObjectUI.RemoteObjectPreviewFormatter();
52 this._searchRegex = null;
53 /** @type {?UI.Icon} */
54 this._messageLevelIcon = null;
55 }
56
57 /**
58 * @override
59 * @return {!Element}
60 */
61 element() {
62 return this.toMessageElement();
63 }
64
65 /**
66 * @return {!Promise<!Element>}
67 */
68 async completeElementForTest() {
69 const element = this.toMessageElement();
70 if (this._completeElementForTestPromise)
71 await this._completeElementForTestPromise;
72 return element;
73 }
74
75 /**
76 * @override
77 */
78 wasShown() {
79 if (this._dataGrid)
80 this._dataGrid.updateWidths();
81 this._isVisible = true;
82 }
83
84 onResize() {
85 if (!this._isVisible)
86 return;
87 if (this._dataGrid)
88 this._dataGrid.onResize();
89 }
90
91 /**
92 * @override
93 */
94 willHide() {
95 this._isVisible = false;
Erik Luo4b002322018-07-30 21:23:3196 this._cachedHeight = this.element().offsetHeight;
Blink Reformat4c46d092018-04-07 15:32:3797 }
98
99 /**
100 * @return {number}
101 */
102 fastHeight() {
103 if (this._cachedHeight)
104 return this._cachedHeight;
105 // This value reflects the 18px min-height of .console-message, plus the
106 // 1px border of .console-message-wrapper. Keep in sync with consoleView.css.
107 const defaultConsoleRowHeight = 19;
108 if (this._message.type === SDK.ConsoleMessage.MessageType.Table) {
109 const table = this._message.parameters[0];
110 if (table && table.preview)
111 return defaultConsoleRowHeight * table.preview.properties.length;
112 }
113 return defaultConsoleRowHeight;
114 }
115
116 /**
117 * @return {!SDK.ConsoleMessage}
118 */
119 consoleMessage() {
120 return this._message;
121 }
122
123 /**
124 * @return {!Element}
125 */
126 _buildTableMessage() {
127 const formattedMessage = createElementWithClass('span', 'source-code');
Erik Luo5976c8c2018-07-24 02:03:09128 this._anchorElement = this._buildMessageAnchor();
129 if (this._anchorElement)
130 formattedMessage.appendChild(this._anchorElement);
Blink Reformat4c46d092018-04-07 15:32:37131 const badgeElement = this._buildMessageBadge();
132 if (badgeElement)
133 formattedMessage.appendChild(badgeElement);
134
135 let table = this._message.parameters && this._message.parameters.length ? this._message.parameters[0] : null;
136 if (table)
137 table = this._parameterToRemoteObject(table);
138 if (!table || !table.preview)
139 return formattedMessage;
140
141 const rawValueColumnSymbol = Symbol('rawValueColumn');
142 const columnNames = [];
143 const preview = table.preview;
144 const rows = [];
145 for (let i = 0; i < preview.properties.length; ++i) {
146 const rowProperty = preview.properties[i];
147 let rowSubProperties;
148 if (rowProperty.valuePreview)
149 rowSubProperties = rowProperty.valuePreview.properties;
150 else if (rowProperty.value)
151 rowSubProperties = [{name: rawValueColumnSymbol, type: rowProperty.type, value: rowProperty.value}];
152 else
153 continue;
154
155 const rowValue = {};
156 const maxColumnsToRender = 20;
157 for (let j = 0; j < rowSubProperties.length; ++j) {
158 const cellProperty = rowSubProperties[j];
159 let columnRendered = columnNames.indexOf(cellProperty.name) !== -1;
160 if (!columnRendered) {
161 if (columnNames.length === maxColumnsToRender)
162 continue;
163 columnRendered = true;
164 columnNames.push(cellProperty.name);
165 }
166
167 if (columnRendered) {
168 const cellElement = this._renderPropertyPreviewOrAccessor(table, [rowProperty, cellProperty]);
169 cellElement.classList.add('console-message-nowrap-below');
170 rowValue[cellProperty.name] = cellElement;
171 }
172 }
173 rows.push([rowProperty.name, rowValue]);
174 }
175
176 const flatValues = [];
177 for (let i = 0; i < rows.length; ++i) {
178 const rowName = rows[i][0];
179 const rowValue = rows[i][1];
180 flatValues.push(rowName);
181 for (let j = 0; j < columnNames.length; ++j)
182 flatValues.push(rowValue[columnNames[j]]);
183 }
184 columnNames.unshift(Common.UIString('(index)'));
185 const columnDisplayNames = columnNames.map(name => name === rawValueColumnSymbol ? Common.UIString('Value') : name);
186
187 if (flatValues.length) {
188 this._dataGrid = DataGrid.SortableDataGrid.create(columnDisplayNames, flatValues);
189 this._dataGrid.setStriped(true);
190
191 const formattedResult = createElementWithClass('span', 'console-message-text');
192 const tableElement = formattedResult.createChild('div', 'console-message-formatted-table');
193 const dataGridContainer = tableElement.createChild('span');
194 tableElement.appendChild(this._formatParameter(table, true, false));
195 dataGridContainer.appendChild(this._dataGrid.element);
196 formattedMessage.appendChild(formattedResult);
197 this._dataGrid.renderInline();
198 }
199 return formattedMessage;
200 }
201
202 /**
203 * @return {!Element}
204 */
205 _buildMessage() {
206 let messageElement;
207 let messageText = this._message.messageText;
208 if (this._message.source === SDK.ConsoleMessage.MessageSource.ConsoleAPI) {
209 switch (this._message.type) {
210 case SDK.ConsoleMessage.MessageType.Trace:
211 messageElement = this._format(this._message.parameters || ['console.trace']);
212 break;
213 case SDK.ConsoleMessage.MessageType.Clear:
214 messageElement = createElementWithClass('span', 'console-info');
215 if (Common.moduleSetting('preserveConsoleLog').get())
216 messageElement.textContent = Common.UIString('console.clear() was prevented due to \'Preserve log\'');
217 else
218 messageElement.textContent = Common.UIString('Console was cleared');
219 messageElement.title =
220 Common.UIString('Clear all messages with ' + UI.shortcutRegistry.shortcutTitleForAction('console.clear'));
221 break;
222 case SDK.ConsoleMessage.MessageType.Assert: {
223 let args = [Common.UIString('Assertion failed:')];
224 if (this._message.parameters)
225 args = args.concat(this._message.parameters);
226 messageElement = this._format(args);
227 break;
228 }
229 case SDK.ConsoleMessage.MessageType.Dir: {
230 const obj = this._message.parameters ? this._message.parameters[0] : undefined;
231 const args = ['%O', obj];
232 messageElement = this._format(args);
233 break;
234 }
235 case SDK.ConsoleMessage.MessageType.Profile:
236 case SDK.ConsoleMessage.MessageType.ProfileEnd:
237 messageElement = this._format([messageText]);
238 break;
239 default: {
240 if (this._message.parameters && this._message.parameters.length === 1 &&
241 this._message.parameters[0].type === 'string')
242 messageElement = this._tryFormatAsError(/** @type {string} */ (this._message.parameters[0].value));
243 const args = this._message.parameters || [messageText];
244 messageElement = messageElement || this._format(args);
245 }
246 }
247 } else {
248 let rendered = false;
249 this._completeElementForTestPromise = null;
250 for (const extension of self.runtime.extensions(Common.Renderer, this._message)) {
251 if (extension.descriptor()['source'] === this._message.source) {
252 messageElement = createElement('span');
253 let callback;
254 this._completeElementForTestPromise = new Promise(fulfill => callback = fulfill);
255 extension.instance().then(renderer => {
256 renderer.render(this._message)
257 .then(element => messageElement.appendChild(element || this._format([messageText])))
258 .then(callback);
259 });
260 rendered = true;
261 break;
262 }
263 }
264 if (!rendered) {
265 const messageInParameters =
266 this._message.parameters && messageText === /** @type {string} */ (this._message.parameters[0]);
267 if (this._message.source === SDK.ConsoleMessage.MessageSource.Violation)
268 messageText = Common.UIString('[Violation] %s', messageText);
269 else if (this._message.source === SDK.ConsoleMessage.MessageSource.Intervention)
270 messageText = Common.UIString('[Intervention] %s', messageText);
271 else if (this._message.source === SDK.ConsoleMessage.MessageSource.Deprecation)
272 messageText = Common.UIString('[Deprecation] %s', messageText);
273 const args = this._message.parameters || [messageText];
274 if (messageInParameters)
275 args[0] = messageText;
276 messageElement = this._format(args);
277 }
278 }
279 messageElement.classList.add('console-message-text');
280
281 const formattedMessage = createElementWithClass('span', 'source-code');
Erik Luo5976c8c2018-07-24 02:03:09282 this._anchorElement = this._buildMessageAnchor();
283 if (this._anchorElement)
284 formattedMessage.appendChild(this._anchorElement);
Blink Reformat4c46d092018-04-07 15:32:37285 const badgeElement = this._buildMessageBadge();
286 if (badgeElement)
287 formattedMessage.appendChild(badgeElement);
288 formattedMessage.appendChild(messageElement);
289 return formattedMessage;
290 }
291
292 /**
293 * @return {?Element}
294 */
295 _buildMessageAnchor() {
296 let anchorElement = null;
297 if (this._message.scriptId) {
298 anchorElement = this._linkifyScriptId(
299 this._message.scriptId, this._message.url || '', this._message.line, this._message.column);
300 } else if (this._message.stackTrace && this._message.stackTrace.callFrames.length) {
301 anchorElement = this._linkifyStackTraceTopFrame(this._message.stackTrace);
302 } else if (this._message.url && this._message.url !== 'undefined') {
303 anchorElement = this._linkifyLocation(this._message.url, this._message.line, this._message.column);
304 }
305
306 // Append a space to prevent the anchor text from being glued to the console message when the user selects and copies the console messages.
307 if (anchorElement) {
308 const anchorWrapperElement = createElementWithClass('span', 'console-message-anchor');
309 anchorWrapperElement.appendChild(anchorElement);
310 anchorWrapperElement.createTextChild(' ');
311 return anchorWrapperElement;
312 }
313 return null;
314 }
315
316 /**
317 * @return {?Element}
318 */
319 _buildMessageBadge() {
320 const badgeElement = this._badgeElement();
321 if (!badgeElement)
322 return null;
323 badgeElement.classList.add('console-message-badge');
324 return badgeElement;
325 }
326
327 /**
328 * @return {?Element}
329 */
330 _badgeElement() {
331 if (this._message._url)
332 return this._badgePool.badgeForURL(new Common.ParsedURL(this._message._url));
333 if (this._message.stackTrace) {
334 let stackTrace = this._message.stackTrace;
335 while (stackTrace) {
336 for (const callFrame of this._message.stackTrace.callFrames) {
337 if (callFrame.url)
338 return this._badgePool.badgeForURL(new Common.ParsedURL(callFrame.url));
339 }
340 stackTrace = stackTrace.parent;
341 }
342 }
343 if (!this._message.executionContextId)
344 return null;
345 const runtimeModel = this._message.runtimeModel();
346 if (!runtimeModel)
347 return null;
348 const executionContext = runtimeModel.executionContext(this._message.executionContextId);
349 if (!executionContext || !executionContext.frameId)
350 return null;
351 const resourceTreeModel = executionContext.target().model(SDK.ResourceTreeModel);
352 if (!resourceTreeModel)
353 return null;
354 const frame = resourceTreeModel.frameForId(executionContext.frameId);
355 if (!frame || !frame.parentFrame)
356 return null;
357 return this._badgePool.badgeForFrame(frame);
358 }
359
360 /**
361 * @return {!Element}
362 */
363 _buildMessageWithStackTrace() {
364 const toggleElement = createElementWithClass('div', 'console-message-stack-trace-toggle');
365 const contentElement = toggleElement.createChild('div', 'console-message-stack-trace-wrapper');
366
367 const messageElement = this._buildMessage();
368 const icon = UI.Icon.create('smallicon-triangle-right', 'console-message-expand-icon');
369 const clickableElement = contentElement.createChild('div');
370 clickableElement.appendChild(icon);
371
372 clickableElement.appendChild(messageElement);
373 const stackTraceElement = contentElement.createChild('div');
374 const stackTracePreview = Components.JSPresentationUtils.buildStackTracePreviewContents(
375 this._message.runtimeModel().target(), this._linkifier, this._message.stackTrace);
376 stackTraceElement.appendChild(stackTracePreview);
377 stackTraceElement.classList.add('hidden');
378
379 /**
380 * @param {boolean} expand
381 */
382 function expandStackTrace(expand) {
383 icon.setIconType(expand ? 'smallicon-triangle-down' : 'smallicon-triangle-right');
384 stackTraceElement.classList.toggle('hidden', !expand);
385 }
386
387 /**
388 * @param {?Event} event
389 */
390 function toggleStackTrace(event) {
391 if (event.target.hasSelection())
392 return;
393 expandStackTrace(stackTraceElement.classList.contains('hidden'));
394 event.consume();
395 }
396
397 clickableElement.addEventListener('click', toggleStackTrace, false);
398 if (this._message.type === SDK.ConsoleMessage.MessageType.Trace)
399 expandStackTrace(true);
400
401 toggleElement._expandStackTraceForTest = expandStackTrace.bind(null, true);
402 return toggleElement;
403 }
404
405 /**
406 * @param {string} url
407 * @param {number} lineNumber
408 * @param {number} columnNumber
409 * @return {?Element}
410 */
411 _linkifyLocation(url, lineNumber, columnNumber) {
412 if (!this._message.runtimeModel())
413 return null;
414 return this._linkifier.linkifyScriptLocation(
415 this._message.runtimeModel().target(), null, url, lineNumber, columnNumber);
416 }
417
418 /**
419 * @param {!Protocol.Runtime.StackTrace} stackTrace
420 * @return {?Element}
421 */
422 _linkifyStackTraceTopFrame(stackTrace) {
423 if (!this._message.runtimeModel())
424 return null;
425 return this._linkifier.linkifyStackTraceTopFrame(this._message.runtimeModel().target(), stackTrace);
426 }
427
428 /**
429 * @param {string} scriptId
430 * @param {string} url
431 * @param {number} lineNumber
432 * @param {number} columnNumber
433 * @return {?Element}
434 */
435 _linkifyScriptId(scriptId, url, lineNumber, columnNumber) {
436 if (!this._message.runtimeModel())
437 return null;
438 return this._linkifier.linkifyScriptLocation(
439 this._message.runtimeModel().target(), scriptId, url, lineNumber, columnNumber);
440 }
441
442 /**
443 * @param {!SDK.RemoteObject|!Protocol.Runtime.RemoteObject|string} parameter
444 * @return {!SDK.RemoteObject}
445 */
446 _parameterToRemoteObject(parameter) {
447 if (parameter instanceof SDK.RemoteObject)
448 return parameter;
449 const runtimeModel = this._message.runtimeModel();
450 if (!runtimeModel)
451 return SDK.RemoteObject.fromLocalObject(parameter);
452 if (typeof parameter === 'object')
453 return runtimeModel.createRemoteObject(parameter);
454 return runtimeModel.createRemoteObjectFromPrimitiveValue(parameter);
455 }
456
457 /**
458 * @param {!Array.<!SDK.RemoteObject|string>} rawParameters
459 * @return {!Element}
460 */
461 _format(rawParameters) {
462 // This node is used like a Builder. Values are continually appended onto it.
463 const formattedResult = createElement('span');
464 if (!rawParameters.length)
465 return formattedResult;
466
467 // Formatting code below assumes that parameters are all wrappers whereas frontend console
468 // API allows passing arbitrary values as messages (strings, numbers, etc.). Wrap them here.
469 // FIXME: Only pass runtime wrappers here.
470 let parameters = [];
471 for (let i = 0; i < rawParameters.length; ++i)
472 parameters[i] = this._parameterToRemoteObject(rawParameters[i]);
473
474 // There can be string log and string eval result. We distinguish between them based on message type.
475 const shouldFormatMessage =
476 SDK.RemoteObject.type((/** @type {!Array.<!SDK.RemoteObject>} **/ (parameters))[0]) === 'string' &&
477 (this._message.type !== SDK.ConsoleMessage.MessageType.Result ||
478 this._message.level === SDK.ConsoleMessage.MessageLevel.Error);
479
480 // Multiple parameters with the first being a format string. Save unused substitutions.
481 if (shouldFormatMessage) {
482 const result = this._formatWithSubstitutionString(
483 /** @type {string} **/ (parameters[0].description), parameters.slice(1), formattedResult);
484 parameters = result.unusedSubstitutions;
485 if (parameters.length)
486 formattedResult.createTextChild(' ');
487 }
488
489 // Single parameter, or unused substitutions from above.
490 for (let i = 0; i < parameters.length; ++i) {
491 // Inline strings when formatting.
492 if (shouldFormatMessage && parameters[i].type === 'string')
493 formattedResult.appendChild(Console.ConsoleViewMessage._linkifyStringAsFragment(parameters[i].description));
494 else
495 formattedResult.appendChild(this._formatParameter(parameters[i], false, true));
496 if (i < parameters.length - 1)
497 formattedResult.createTextChild(' ');
498 }
499 return formattedResult;
500 }
501
502 /**
503 * @param {!SDK.RemoteObject} output
504 * @param {boolean=} forceObjectFormat
505 * @param {boolean=} includePreview
506 * @return {!Element}
507 */
508 _formatParameter(output, forceObjectFormat, includePreview) {
509 if (output.customPreview())
510 return (new ObjectUI.CustomPreviewComponent(output)).element;
511
512 const type = forceObjectFormat ? 'object' : (output.subtype || output.type);
513 let element;
514 switch (type) {
515 case 'error':
516 element = this._formatParameterAsError(output);
517 break;
518 case 'function':
519 element = this._formatParameterAsFunction(output, includePreview);
520 break;
521 case 'array':
522 case 'arraybuffer':
523 case 'blob':
524 case 'dataview':
525 case 'generator':
526 case 'iterator':
527 case 'map':
528 case 'object':
529 case 'promise':
530 case 'proxy':
531 case 'set':
532 case 'typedarray':
533 case 'weakmap':
534 case 'weakset':
535 element = this._formatParameterAsObject(output, includePreview);
536 break;
537 case 'node':
538 element = output.isNode() ? this._formatParameterAsNode(output) : this._formatParameterAsObject(output, false);
539 break;
540 case 'string':
541 element = this._formatParameterAsString(output);
542 break;
543 case 'boolean':
544 case 'date':
545 case 'null':
546 case 'number':
547 case 'regexp':
548 case 'symbol':
549 case 'undefined':
550 case 'bigint':
551 element = this._formatParameterAsValue(output);
552 break;
553 default:
554 element = this._formatParameterAsValue(output);
555 console.error('Tried to format remote object of unknown type.');
556 }
557 element.classList.add('object-value-' + type);
558 element.classList.add('source-code');
559 return element;
560 }
561
562 /**
563 * @param {!SDK.RemoteObject} obj
564 * @return {!Element}
565 */
566 _formatParameterAsValue(obj) {
567 const result = createElement('span');
568 const description = obj.description || '';
569 if (description.length > Console.ConsoleViewMessage._MaxTokenizableStringLength)
Erik Luo5fb33bd2018-06-04 23:23:52570 result.appendChild(UI.createExpandableText(description, Console.ConsoleViewMessage._LongStringVisibleLength));
Blink Reformat4c46d092018-04-07 15:32:37571 else
572 result.createTextChild(description);
573 if (obj.objectId)
574 result.addEventListener('contextmenu', this._contextMenuEventFired.bind(this, obj), false);
575 return result;
576 }
577
578 /**
579 * @param {!SDK.RemoteObject} obj
580 * @param {boolean=} includePreview
581 * @return {!Element}
582 */
583 _formatParameterAsObject(obj, includePreview) {
584 const titleElement = createElementWithClass('span', 'console-object');
585 if (includePreview && obj.preview) {
586 titleElement.classList.add('console-object-preview');
587 this._previewFormatter.appendObjectPreview(titleElement, obj.preview, false /* isEntry */);
588 } else if (obj.type === 'function') {
589 const functionElement = titleElement.createChild('span');
590 ObjectUI.ObjectPropertiesSection.formatObjectAsFunction(obj, functionElement, false);
591 titleElement.classList.add('object-value-function');
592 } else {
593 titleElement.createTextChild(obj.description || '');
594 }
595
596 if (!obj.hasChildren || obj.customPreview())
597 return titleElement;
598
599 const note = titleElement.createChild('span', 'object-state-note info-note');
600 if (this._message.type === SDK.ConsoleMessage.MessageType.QueryObjectResult)
601 note.title = ls`This value will not be collected until console is cleared.`;
602 else
603 note.title = ls`Value below was evaluated just now.`;
604
605 const section = new ObjectUI.ObjectPropertiesSection(obj, titleElement, this._linkifier);
606 section.element.classList.add('console-view-object-properties-section');
607 section.enableContextMenu();
608 return section.element;
609 }
610
611 /**
612 * @param {!SDK.RemoteObject} func
613 * @param {boolean=} includePreview
614 * @return {!Element}
615 */
616 _formatParameterAsFunction(func, includePreview) {
617 const result = createElement('span');
618 SDK.RemoteFunction.objectAsFunction(func).targetFunction().then(formatTargetFunction.bind(this));
619 return result;
620
621 /**
622 * @param {!SDK.RemoteObject} targetFunction
623 * @this {Console.ConsoleViewMessage}
624 */
625 function formatTargetFunction(targetFunction) {
626 const functionElement = createElement('span');
627 ObjectUI.ObjectPropertiesSection.formatObjectAsFunction(targetFunction, functionElement, true, includePreview);
628 result.appendChild(functionElement);
629 if (targetFunction !== func) {
630 const note = result.createChild('span', 'object-info-state-note');
631 note.title = Common.UIString('Function was resolved from bound function.');
632 }
633 result.addEventListener('contextmenu', this._contextMenuEventFired.bind(this, targetFunction), false);
634 }
635 }
636
637 /**
638 * @param {!SDK.RemoteObject} obj
639 * @param {!Event} event
640 */
641 _contextMenuEventFired(obj, event) {
642 const contextMenu = new UI.ContextMenu(event);
643 contextMenu.appendApplicableItems(obj);
644 contextMenu.show();
645 }
646
647 /**
648 * @param {?SDK.RemoteObject} object
649 * @param {!Array.<!Protocol.Runtime.PropertyPreview>} propertyPath
650 * @return {!Element}
651 */
652 _renderPropertyPreviewOrAccessor(object, propertyPath) {
653 const property = propertyPath.peekLast();
654 if (property.type === 'accessor')
655 return this._formatAsAccessorProperty(object, propertyPath.map(property => property.name), false);
656 return this._previewFormatter.renderPropertyPreview(
657 property.type, /** @type {string} */ (property.subtype), property.value);
658 }
659
660 /**
661 * @param {!SDK.RemoteObject} remoteObject
662 * @return {!Element}
663 */
664 _formatParameterAsNode(remoteObject) {
665 const result = createElement('span');
666
667 const domModel = remoteObject.runtimeModel().target().model(SDK.DOMModel);
668 if (!domModel)
669 return result;
670 domModel.pushObjectAsNodeToFrontend(remoteObject).then(node => {
671 if (!node) {
672 result.appendChild(this._formatParameterAsObject(remoteObject, false));
673 return;
674 }
675 Common.Renderer.render(node).then(rendererNode => {
676 if (rendererNode)
677 result.appendChild(rendererNode);
678 else
679 result.appendChild(this._formatParameterAsObject(remoteObject, false));
680 this._formattedParameterAsNodeForTest();
681 });
682 });
683
684 return result;
685 }
686
687 _formattedParameterAsNodeForTest() {
688 }
689
690 /**
691 * @param {!SDK.RemoteObject} output
692 * @return {!Element}
693 */
694 _formatParameterAsString(output) {
695 const span = createElement('span');
696 span.appendChild(Console.ConsoleViewMessage._linkifyStringAsFragment(output.description || ''));
697
698 const result = createElement('span');
699 result.createChild('span', 'object-value-string-quote').textContent = '"';
700 result.appendChild(span);
701 result.createChild('span', 'object-value-string-quote').textContent = '"';
702 return result;
703 }
704
705 /**
706 * @param {!SDK.RemoteObject} output
707 * @return {!Element}
708 */
709 _formatParameterAsError(output) {
710 const result = createElement('span');
711 const errorSpan = this._tryFormatAsError(output.description || '');
712 result.appendChild(
713 errorSpan ? errorSpan : Console.ConsoleViewMessage._linkifyStringAsFragment(output.description || ''));
714 return result;
715 }
716
717 /**
718 * @param {!SDK.RemoteObject} output
719 * @return {!Element}
720 */
721 _formatAsArrayEntry(output) {
722 return this._previewFormatter.renderPropertyPreview(output.type, output.subtype, output.description);
723 }
724
725 /**
726 * @param {?SDK.RemoteObject} object
727 * @param {!Array.<string>} propertyPath
728 * @param {boolean} isArrayEntry
729 * @return {!Element}
730 */
731 _formatAsAccessorProperty(object, propertyPath, isArrayEntry) {
732 const rootElement = ObjectUI.ObjectPropertyTreeElement.createRemoteObjectAccessorPropertySpan(
733 object, propertyPath, onInvokeGetterClick.bind(this));
734
735 /**
736 * @param {?SDK.RemoteObject} result
737 * @param {boolean=} wasThrown
738 * @this {Console.ConsoleViewMessage}
739 */
740 function onInvokeGetterClick(result, wasThrown) {
741 if (!result)
742 return;
743 rootElement.removeChildren();
744 if (wasThrown) {
745 const element = rootElement.createChild('span');
746 element.textContent = Common.UIString('<exception>');
747 element.title = /** @type {string} */ (result.description);
748 } else if (isArrayEntry) {
749 rootElement.appendChild(this._formatAsArrayEntry(result));
750 } else {
751 // Make a PropertyPreview from the RemoteObject similar to the backend logic.
752 const maxLength = 100;
753 const type = result.type;
754 const subtype = result.subtype;
755 let description = '';
756 if (type !== 'function' && result.description) {
757 if (type === 'string' || subtype === 'regexp')
758 description = result.description.trimMiddle(maxLength);
759 else
760 description = result.description.trimEnd(maxLength);
761 }
762 rootElement.appendChild(this._previewFormatter.renderPropertyPreview(type, subtype, description));
763 }
764 }
765
766 return rootElement;
767 }
768
769 /**
770 * @param {string} format
771 * @param {!Array.<!SDK.RemoteObject>} parameters
772 * @param {!Element} formattedResult
773 */
774 _formatWithSubstitutionString(format, parameters, formattedResult) {
775 const formatters = {};
776
777 /**
778 * @param {boolean} force
779 * @param {boolean} includePreview
780 * @param {!SDK.RemoteObject} obj
781 * @return {!Element}
782 * @this {Console.ConsoleViewMessage}
783 */
784 function parameterFormatter(force, includePreview, obj) {
785 return this._formatParameter(obj, force, includePreview);
786 }
787
788 function stringFormatter(obj) {
789 return obj.description;
790 }
791
792 function floatFormatter(obj) {
793 if (typeof obj.value !== 'number')
794 return 'NaN';
795 return obj.value;
796 }
797
798 function integerFormatter(obj) {
799 if (typeof obj.value !== 'number')
800 return 'NaN';
801 return Math.floor(obj.value);
802 }
803
804 function bypassFormatter(obj) {
805 return (obj instanceof Node) ? obj : '';
806 }
807
808 let currentStyle = null;
809 function styleFormatter(obj) {
810 currentStyle = {};
811 const buffer = createElement('span');
812 buffer.setAttribute('style', obj.description);
813 for (let i = 0; i < buffer.style.length; i++) {
814 const property = buffer.style[i];
815 if (isWhitelistedProperty(property))
816 currentStyle[property] = buffer.style[property];
817 }
818 }
819
820 function isWhitelistedProperty(property) {
821 // Make sure that allowed properties do not interfere with link visibility.
822 const prefixes = [
823 'background', 'border', 'color', 'font', 'line', 'margin', 'padding', 'text', '-webkit-background',
824 '-webkit-border', '-webkit-font', '-webkit-margin', '-webkit-padding', '-webkit-text'
825 ];
826 for (let i = 0; i < prefixes.length; i++) {
827 if (property.startsWith(prefixes[i]))
828 return true;
829 }
830 return false;
831 }
832
833 // Firebug uses %o for formatting objects.
834 formatters.o = parameterFormatter.bind(this, false /* force */, true /* includePreview */);
835 formatters.s = stringFormatter;
836 formatters.f = floatFormatter;
837 // Firebug allows both %i and %d for formatting integers.
838 formatters.i = integerFormatter;
839 formatters.d = integerFormatter;
840
841 // Firebug uses %c for styling the message.
842 formatters.c = styleFormatter;
843
844 // Support %O to force object formatting, instead of the type-based %o formatting.
845 formatters.O = parameterFormatter.bind(this, true /* force */, false /* includePreview */);
846
847 formatters._ = bypassFormatter;
848
849 /**
850 * @param {!Element} a
851 * @param {*} b
852 * @this {!Console.ConsoleViewMessage}
Erik Luo17926392018-05-17 22:06:12853 * @return {!Element}
Blink Reformat4c46d092018-04-07 15:32:37854 */
855 function append(a, b) {
856 if (b instanceof Node) {
857 a.appendChild(b);
Erik Luo17926392018-05-17 22:06:12858 return a;
859 }
860 if (typeof b === 'undefined')
861 return a;
862 if (!currentStyle) {
863 a.appendChild(Console.ConsoleViewMessage._linkifyStringAsFragment(String(b)));
864 return a;
865 }
866 const lines = String(b).split('\n');
867 for (let i = 0; i < lines.length; i++) {
868 const line = lines[i];
869 const lineFragment = Console.ConsoleViewMessage._linkifyStringAsFragment(line);
870 const wrapper = createElement('span');
871 wrapper.style.setProperty('contain', 'paint');
872 wrapper.style.setProperty('display', 'inline-block');
873 wrapper.style.setProperty('max-width', '100%');
874 wrapper.appendChild(lineFragment);
875 applyCurrentStyle(wrapper);
876 for (const child of wrapper.children) {
877 if (child.classList.contains('devtools-link'))
878 this._applyForcedVisibleStyle(child);
Blink Reformat4c46d092018-04-07 15:32:37879 }
Erik Luo17926392018-05-17 22:06:12880 a.appendChild(wrapper);
881 if (i < lines.length - 1)
882 a.appendChild(createElement('br'));
Blink Reformat4c46d092018-04-07 15:32:37883 }
884 return a;
885 }
886
887 /**
888 * @param {!Element} element
889 */
890 function applyCurrentStyle(element) {
891 for (const key in currentStyle)
892 element.style[key] = currentStyle[key];
893 }
894
895 // String.format does treat formattedResult like a Builder, result is an object.
896 return String.format(format, parameters, formatters, formattedResult, append.bind(this));
897 }
898
899 /**
900 * @param {!Element} element
901 */
902 _applyForcedVisibleStyle(element) {
903 element.style.setProperty('-webkit-text-stroke', '0', 'important');
904 element.style.setProperty('text-decoration', 'underline', 'important');
905
906 const themedColor = UI.themeSupport.patchColorText('rgb(33%, 33%, 33%)', UI.ThemeSupport.ColorUsage.Foreground);
907 element.style.setProperty('color', themedColor, 'important');
908
909 let backgroundColor = 'hsl(0, 0%, 100%)';
910 if (this._message.level === SDK.ConsoleMessage.MessageLevel.Error)
911 backgroundColor = 'hsl(0, 100%, 97%)';
912 else if (this._message.level === SDK.ConsoleMessage.MessageLevel.Warning || this._shouldRenderAsWarning())
913 backgroundColor = 'hsl(50, 100%, 95%)';
914 const themedBackgroundColor =
915 UI.themeSupport.patchColorText(backgroundColor, UI.ThemeSupport.ColorUsage.Background);
916 element.style.setProperty('background-color', themedBackgroundColor, 'important');
917 }
918
919 /**
920 * @return {boolean}
921 */
922 matchesFilterRegex(regexObject) {
923 regexObject.lastIndex = 0;
Erik Luo5976c8c2018-07-24 02:03:09924 const contentElement = this.contentElement();
925 const anchorText = this._anchorElement ? this._anchorElement.deepTextContent() : '';
926 return (anchorText && regexObject.test(anchorText.trim())) ||
927 regexObject.test(contentElement.deepTextContent().slice(anchorText.length));
Blink Reformat4c46d092018-04-07 15:32:37928 }
929
930 /**
931 * @param {string} filter
932 * @return {boolean}
933 */
934 matchesFilterText(filter) {
935 const text = this.contentElement().deepTextContent();
936 return text.toLowerCase().includes(filter.toLowerCase());
937 }
938
939 updateTimestamp() {
940 if (!this._contentElement)
941 return;
942
943 if (Common.moduleSetting('consoleTimestampsEnabled').get()) {
944 if (!this._timestampElement)
945 this._timestampElement = createElementWithClass('span', 'console-timestamp');
946 this._timestampElement.textContent = formatTimestamp(this._message.timestamp, false) + ' ';
947 this._timestampElement.title = formatTimestamp(this._message.timestamp, true);
948 this._contentElement.insertBefore(this._timestampElement, this._contentElement.firstChild);
949 } else if (this._timestampElement) {
950 this._timestampElement.remove();
951 delete this._timestampElement;
952 }
953
954 /**
955 * @param {number} timestamp
956 * @param {boolean} full
957 * @return {string}
958 */
959 function formatTimestamp(timestamp, full) {
960 const date = new Date(timestamp);
961 const yymmdd = date.getFullYear() + '-' + leadZero(date.getMonth() + 1, 2) + '-' + leadZero(date.getDate(), 2);
962 const hhmmssfff = leadZero(date.getHours(), 2) + ':' + leadZero(date.getMinutes(), 2) + ':' +
963 leadZero(date.getSeconds(), 2) + '.' + leadZero(date.getMilliseconds(), 3);
964 return full ? (yymmdd + ' ' + hhmmssfff) : hhmmssfff;
965
966 /**
967 * @param {number} value
968 * @param {number} length
969 * @return {string}
970 */
971 function leadZero(value, length) {
972 const valueString = value.toString();
973 const padding = length - valueString.length;
974 return padding <= 0 ? valueString : '0'.repeat(padding) + valueString;
975 }
976 }
977 }
978
979 /**
980 * @return {number}
981 */
982 nestingLevel() {
983 return this._nestingLevel;
984 }
985
986 /**
987 * @param {boolean} inSimilarGroup
988 * @param {boolean=} isLast
989 */
990 setInSimilarGroup(inSimilarGroup, isLast) {
991 this._inSimilarGroup = inSimilarGroup;
992 this._lastInSimilarGroup = inSimilarGroup && !!isLast;
993 if (this._similarGroupMarker && !inSimilarGroup) {
994 this._similarGroupMarker.remove();
995 this._similarGroupMarker = null;
996 } else if (this._element && !this._similarGroupMarker && inSimilarGroup) {
997 this._similarGroupMarker = createElementWithClass('div', 'nesting-level-marker');
998 this._element.insertBefore(this._similarGroupMarker, this._element.firstChild);
999 this._similarGroupMarker.classList.toggle('group-closed', this._lastInSimilarGroup);
1000 }
1001 }
1002
1003 /**
1004 * @return {boolean}
1005 */
1006 isLastInSimilarGroup() {
1007 return this._inSimilarGroup && this._lastInSimilarGroup;
1008 }
1009
1010 resetCloseGroupDecorationCount() {
1011 if (!this._closeGroupDecorationCount)
1012 return;
1013 this._closeGroupDecorationCount = 0;
1014 this._updateCloseGroupDecorations();
1015 }
1016
1017 incrementCloseGroupDecorationCount() {
1018 ++this._closeGroupDecorationCount;
1019 this._updateCloseGroupDecorations();
1020 }
1021
1022 _updateCloseGroupDecorations() {
1023 if (!this._nestingLevelMarkers)
1024 return;
1025 for (let i = 0, n = this._nestingLevelMarkers.length; i < n; ++i) {
1026 const marker = this._nestingLevelMarkers[i];
1027 marker.classList.toggle('group-closed', n - i <= this._closeGroupDecorationCount);
1028 }
1029 }
1030
1031 /**
1032 * @return {!Element}
1033 */
1034 contentElement() {
1035 if (this._contentElement)
1036 return this._contentElement;
1037
1038 const contentElement = createElementWithClass('div', 'console-message');
1039 if (this._messageLevelIcon)
1040 contentElement.appendChild(this._messageLevelIcon);
1041 this._contentElement = contentElement;
1042
1043 let formattedMessage;
1044 const shouldIncludeTrace = !!this._message.stackTrace &&
1045 (this._message.source === SDK.ConsoleMessage.MessageSource.Network ||
1046 this._message.source === SDK.ConsoleMessage.MessageSource.Violation ||
1047 this._message.level === SDK.ConsoleMessage.MessageLevel.Error ||
1048 this._message.level === SDK.ConsoleMessage.MessageLevel.Warning ||
1049 this._message.type === SDK.ConsoleMessage.MessageType.Trace);
1050 if (this._message.runtimeModel() && shouldIncludeTrace)
1051 formattedMessage = this._buildMessageWithStackTrace();
1052 else if (this._message.type === SDK.ConsoleMessage.MessageType.Table)
1053 formattedMessage = this._buildTableMessage();
1054 else
1055 formattedMessage = this._buildMessage();
1056 contentElement.appendChild(formattedMessage);
1057
1058 this.updateTimestamp();
1059 return this._contentElement;
1060 }
1061
1062 /**
1063 * @return {!Element}
1064 */
1065 toMessageElement() {
1066 if (this._element)
1067 return this._element;
1068
1069 this._element = createElement('div');
1070 this.updateMessageElement();
1071 return this._element;
1072 }
1073
1074 updateMessageElement() {
1075 if (!this._element)
1076 return;
1077
1078 this._element.className = 'console-message-wrapper';
1079 this._element.removeChildren();
1080 if (this._message.isGroupStartMessage())
1081 this._element.classList.add('console-group-title');
1082 if (this._message.source === SDK.ConsoleMessage.MessageSource.ConsoleAPI)
1083 this._element.classList.add('console-from-api');
1084 if (this._inSimilarGroup) {
1085 this._similarGroupMarker = this._element.createChild('div', 'nesting-level-marker');
1086 this._similarGroupMarker.classList.toggle('group-closed', this._lastInSimilarGroup);
1087 }
1088
1089 this._nestingLevelMarkers = [];
1090 for (let i = 0; i < this._nestingLevel; ++i)
1091 this._nestingLevelMarkers.push(this._element.createChild('div', 'nesting-level-marker'));
1092 this._updateCloseGroupDecorations();
1093 this._element.message = this;
1094
1095 switch (this._message.level) {
1096 case SDK.ConsoleMessage.MessageLevel.Verbose:
1097 this._element.classList.add('console-verbose-level');
1098 this._updateMessageLevelIcon('');
1099 break;
1100 case SDK.ConsoleMessage.MessageLevel.Info:
1101 this._element.classList.add('console-info-level');
1102 if (this._message.type === SDK.ConsoleMessage.MessageType.System)
1103 this._element.classList.add('console-system-type');
1104 break;
1105 case SDK.ConsoleMessage.MessageLevel.Warning:
1106 this._element.classList.add('console-warning-level');
1107 this._updateMessageLevelIcon('smallicon-warning');
1108 break;
1109 case SDK.ConsoleMessage.MessageLevel.Error:
1110 this._element.classList.add('console-error-level');
1111 this._updateMessageLevelIcon('smallicon-error');
1112 break;
1113 }
1114 if (this._shouldRenderAsWarning())
1115 this._element.classList.add('console-warning-level');
1116
1117 this._element.appendChild(this.contentElement());
1118 if (this._repeatCount > 1)
1119 this._showRepeatCountElement();
1120 }
1121
1122 /**
1123 * @return {boolean}
1124 */
1125 _shouldRenderAsWarning() {
1126 return (this._message.level === SDK.ConsoleMessage.MessageLevel.Verbose ||
1127 this._message.level === SDK.ConsoleMessage.MessageLevel.Info) &&
1128 (this._message.source === SDK.ConsoleMessage.MessageSource.Violation ||
1129 this._message.source === SDK.ConsoleMessage.MessageSource.Deprecation ||
1130 this._message.source === SDK.ConsoleMessage.MessageSource.Intervention ||
1131 this._message.source === SDK.ConsoleMessage.MessageSource.Recommendation);
1132 }
1133
1134 /**
1135 * @param {string} iconType
1136 */
1137 _updateMessageLevelIcon(iconType) {
1138 if (!iconType && !this._messageLevelIcon)
1139 return;
1140 if (iconType && !this._messageLevelIcon) {
1141 this._messageLevelIcon = UI.Icon.create('', 'message-level-icon');
1142 if (this._contentElement)
1143 this._contentElement.insertBefore(this._messageLevelIcon, this._contentElement.firstChild);
1144 }
1145 this._messageLevelIcon.setIconType(iconType);
1146 }
1147
1148 /**
1149 * @return {number}
1150 */
1151 repeatCount() {
1152 return this._repeatCount || 1;
1153 }
1154
1155 resetIncrementRepeatCount() {
1156 this._repeatCount = 1;
1157 if (!this._repeatCountElement)
1158 return;
1159
1160 this._repeatCountElement.remove();
1161 if (this._contentElement)
1162 this._contentElement.classList.remove('repeated-message');
1163 delete this._repeatCountElement;
1164 }
1165
1166 incrementRepeatCount() {
1167 this._repeatCount++;
1168 this._showRepeatCountElement();
1169 }
1170
1171 /**
1172 * @param {number} repeatCount
1173 */
1174 setRepeatCount(repeatCount) {
1175 this._repeatCount = repeatCount;
1176 this._showRepeatCountElement();
1177 }
1178
1179 _showRepeatCountElement() {
1180 if (!this._element)
1181 return;
1182
1183 if (!this._repeatCountElement) {
1184 this._repeatCountElement = createElementWithClass('label', 'console-message-repeat-count', 'dt-small-bubble');
1185 switch (this._message.level) {
1186 case SDK.ConsoleMessage.MessageLevel.Warning:
1187 this._repeatCountElement.type = 'warning';
1188 break;
1189 case SDK.ConsoleMessage.MessageLevel.Error:
1190 this._repeatCountElement.type = 'error';
1191 break;
1192 case SDK.ConsoleMessage.MessageLevel.Verbose:
1193 this._repeatCountElement.type = 'verbose';
1194 break;
1195 default:
1196 this._repeatCountElement.type = 'info';
1197 }
1198 if (this._shouldRenderAsWarning())
1199 this._repeatCountElement.type = 'warning';
1200
1201 this._element.insertBefore(this._repeatCountElement, this._contentElement);
1202 this._contentElement.classList.add('repeated-message');
1203 }
1204 this._repeatCountElement.textContent = this._repeatCount;
1205 }
1206
1207 get text() {
1208 return this._message.messageText;
1209 }
1210
1211 /**
1212 * @return {string}
1213 */
1214 toExportString() {
1215 const lines = [];
1216 const nodes = this.contentElement().childTextNodes();
1217 const messageContent = nodes.map(Components.Linkifier.untruncatedNodeText).join('');
1218 for (let i = 0; i < this.repeatCount(); ++i)
1219 lines.push(messageContent);
1220 return lines.join('\n');
1221 }
1222
1223 /**
1224 * @param {?RegExp} regex
1225 */
1226 setSearchRegex(regex) {
1227 if (this._searchHiglightNodeChanges && this._searchHiglightNodeChanges.length)
1228 UI.revertDomChanges(this._searchHiglightNodeChanges);
1229 this._searchRegex = regex;
1230 this._searchHighlightNodes = [];
1231 this._searchHiglightNodeChanges = [];
1232 if (!this._searchRegex)
1233 return;
1234
1235 const text = this.contentElement().deepTextContent();
1236 let match;
1237 this._searchRegex.lastIndex = 0;
1238 const sourceRanges = [];
1239 while ((match = this._searchRegex.exec(text)) && match[0])
1240 sourceRanges.push(new TextUtils.SourceRange(match.index, match[0].length));
1241
1242 if (sourceRanges.length) {
1243 this._searchHighlightNodes =
1244 UI.highlightSearchResults(this.contentElement(), sourceRanges, this._searchHiglightNodeChanges);
1245 }
1246 }
1247
1248 /**
1249 * @return {?RegExp}
1250 */
1251 searchRegex() {
1252 return this._searchRegex;
1253 }
1254
1255 /**
1256 * @return {number}
1257 */
1258 searchCount() {
1259 return this._searchHighlightNodes.length;
1260 }
1261
1262 /**
1263 * @return {!Element}
1264 */
1265 searchHighlightNode(index) {
1266 return this._searchHighlightNodes[index];
1267 }
1268
1269 /**
1270 * @param {string} string
1271 * @return {?Element}
1272 */
1273 _tryFormatAsError(string) {
1274 /**
1275 * @param {string} prefix
1276 */
1277 function startsWith(prefix) {
1278 return string.startsWith(prefix);
1279 }
1280
1281 const errorPrefixes =
1282 ['EvalError', 'ReferenceError', 'SyntaxError', 'TypeError', 'RangeError', 'Error', 'URIError'];
1283 if (!this._message.runtimeModel() || !errorPrefixes.some(startsWith))
1284 return null;
1285 const debuggerModel = this._message.runtimeModel().debuggerModel();
1286 const baseURL = this._message.runtimeModel().target().inspectedURL();
1287
1288 const lines = string.split('\n');
1289 const links = [];
1290 let position = 0;
1291 for (let i = 0; i < lines.length; ++i) {
1292 position += i > 0 ? lines[i - 1].length + 1 : 0;
1293 const isCallFrameLine = /^\s*at\s/.test(lines[i]);
1294 if (!isCallFrameLine && links.length)
1295 return null;
1296
1297 if (!isCallFrameLine)
1298 continue;
1299
1300 let openBracketIndex = -1;
1301 let closeBracketIndex = -1;
1302 const match = /\([^\)\(]+\)/.exec(lines[i]);
1303 if (match) {
1304 openBracketIndex = match.index;
1305 closeBracketIndex = match.index + match[0].length - 1;
1306 }
1307 const hasOpenBracket = openBracketIndex !== -1;
1308 const left = hasOpenBracket ? openBracketIndex + 1 : lines[i].indexOf('at') + 3;
1309 const right = hasOpenBracket ? closeBracketIndex : lines[i].length;
1310 const linkCandidate = lines[i].substring(left, right);
1311 const splitResult = Common.ParsedURL.splitLineAndColumn(linkCandidate);
1312 if (!splitResult)
1313 return null;
1314
1315 if (splitResult.url === '<anonymous>')
1316 continue;
1317 let url = parseOrScriptMatch(splitResult.url);
1318 if (!url && Common.ParsedURL.isRelativeURL(splitResult.url))
1319 url = parseOrScriptMatch(Common.ParsedURL.completeURL(baseURL, splitResult.url));
1320 if (!url)
1321 return null;
1322
1323 links.push({
1324 url: url,
1325 positionLeft: position + left,
1326 positionRight: position + right,
1327 lineNumber: splitResult.lineNumber,
1328 columnNumber: splitResult.columnNumber
1329 });
1330 }
1331
1332 if (!links.length)
1333 return null;
1334
1335 const formattedResult = createElement('span');
1336 let start = 0;
1337 for (let i = 0; i < links.length; ++i) {
1338 formattedResult.appendChild(
1339 Console.ConsoleViewMessage._linkifyStringAsFragment(string.substring(start, links[i].positionLeft)));
1340 formattedResult.appendChild(this._linkifier.linkifyScriptLocation(
1341 debuggerModel.target(), null, links[i].url, links[i].lineNumber, links[i].columnNumber));
1342 start = links[i].positionRight;
1343 }
1344
1345 if (start !== string.length)
1346 formattedResult.appendChild(Console.ConsoleViewMessage._linkifyStringAsFragment(string.substring(start)));
1347
1348 return formattedResult;
1349
1350 /**
1351 * @param {?string} url
1352 * @return {?string}
1353 */
1354 function parseOrScriptMatch(url) {
1355 if (!url)
1356 return null;
1357 const parsedURL = url.asParsedURL();
1358 if (parsedURL)
1359 return parsedURL.url;
1360 if (debuggerModel.scriptsForSourceURL(url).length)
1361 return url;
1362 return null;
1363 }
1364 }
1365
1366 /**
1367 * @param {string} string
1368 * @param {function(string,string,number=,number=):!Node} linkifier
1369 * @return {!DocumentFragment}
1370 */
1371 static linkifyWithCustomLinkifier(string, linkifier) {
1372 if (string.length > Console.ConsoleViewMessage._MaxTokenizableStringLength)
Erik Luo5fb33bd2018-06-04 23:23:521373 return UI.createExpandableText(string, Console.ConsoleViewMessage._LongStringVisibleLength);
Blink Reformat4c46d092018-04-07 15:32:371374 const container = createDocumentFragment();
1375 const tokens = this._tokenizeMessageText(string);
1376 for (const token of tokens) {
1377 switch (token.type) {
1378 case 'url': {
1379 const realURL = (token.text.startsWith('www.') ? 'http://' + token.text : token.text);
1380 const splitResult = Common.ParsedURL.splitLineAndColumn(realURL);
1381 let linkNode;
1382 if (splitResult)
1383 linkNode = linkifier(token.text, splitResult.url, splitResult.lineNumber, splitResult.columnNumber);
1384 else
1385 linkNode = linkifier(token.text, token.value);
1386 container.appendChild(linkNode);
1387 break;
1388 }
1389 default:
1390 container.appendChild(createTextNode(token.text));
1391 break;
1392 }
1393 }
1394 return container;
1395 }
1396
1397 /**
Blink Reformat4c46d092018-04-07 15:32:371398 * @param {string} string
1399 * @return {!DocumentFragment}
1400 */
1401 static _linkifyStringAsFragment(string) {
1402 return Console.ConsoleViewMessage.linkifyWithCustomLinkifier(string, (text, url, lineNumber, columnNumber) => {
1403 return Components.Linkifier.linkifyURL(url, {text, lineNumber, columnNumber});
1404 });
1405 }
1406
1407 /**
1408 * @param {string} string
1409 * @return {!Array<{type: string, text: (string|undefined)}>}
1410 */
1411 static _tokenizeMessageText(string) {
1412 if (!Console.ConsoleViewMessage._tokenizerRegexes) {
1413 const controlCodes = '\\u0000-\\u0020\\u007f-\\u009f';
1414 const linkStringRegex = new RegExp(
1415 '(?:[a-zA-Z][a-zA-Z0-9+.-]{2,}:\\/\\/|data:|www\\.)[^\\s' + controlCodes + '"]{2,}[^\\s' + controlCodes +
1416 '"\')}\\],:;.!?]',
1417 'u');
1418 const pathLineRegex = /(?:\/[\w\.-]*)+\:[\d]+/;
1419 const timeRegex = /took [\d]+ms/;
1420 const eventRegex = /'\w+' event/;
1421 const milestoneRegex = /\sM[6-7]\d/;
1422 const autofillRegex = /\(suggested: \"[\w-]+\"\)/;
1423 const handlers = new Map();
1424 handlers.set(linkStringRegex, 'url');
1425 handlers.set(pathLineRegex, 'url');
1426 handlers.set(timeRegex, 'time');
1427 handlers.set(eventRegex, 'event');
1428 handlers.set(milestoneRegex, 'milestone');
1429 handlers.set(autofillRegex, 'autofill');
1430 Console.ConsoleViewMessage._tokenizerRegexes = Array.from(handlers.keys());
1431 Console.ConsoleViewMessage._tokenizerTypes = Array.from(handlers.values());
1432 }
1433 if (string.length > Console.ConsoleViewMessage._MaxTokenizableStringLength)
1434 return [{text: string, type: undefined}];
1435 const results = TextUtils.TextUtils.splitStringByRegexes(string, Console.ConsoleViewMessage._tokenizerRegexes);
1436 return results.map(
1437 result => ({text: result.value, type: Console.ConsoleViewMessage._tokenizerTypes[result.regexIndex]}));
1438 }
1439
1440 /**
1441 * @return {string}
1442 */
1443 groupKey() {
1444 if (!this._groupKey)
1445 this._groupKey = this._message.groupCategoryKey() + ':' + this.groupTitle();
1446 return this._groupKey;
1447 }
1448
1449 /**
1450 * @return {string}
1451 */
1452 groupTitle() {
1453 const tokens = Console.ConsoleViewMessage._tokenizeMessageText(this._message.messageText);
1454 const result = tokens.reduce((acc, token) => {
1455 let text = token.text;
1456 if (token.type === 'url')
1457 text = Common.UIString('<URL>');
1458 else if (token.type === 'time')
1459 text = Common.UIString('took <N>ms');
1460 else if (token.type === 'event')
1461 text = Common.UIString('<some> event');
1462 else if (token.type === 'milestone')
1463 text = Common.UIString(' M<XX>');
1464 else if (token.type === 'autofill')
1465 text = Common.UIString('<attribute>');
1466 return acc + text;
1467 }, '');
1468 return result.replace(/[%]o/g, '');
1469 }
1470};
1471
1472/**
1473 * @unrestricted
1474 */
1475Console.ConsoleGroupViewMessage = class extends Console.ConsoleViewMessage {
1476 /**
1477 * @param {!SDK.ConsoleMessage} consoleMessage
1478 * @param {!Components.Linkifier} linkifier
1479 * @param {!ProductRegistry.BadgePool} badgePool
1480 * @param {number} nestingLevel
1481 */
1482 constructor(consoleMessage, linkifier, badgePool, nestingLevel) {
1483 console.assert(consoleMessage.isGroupStartMessage());
1484 super(consoleMessage, linkifier, badgePool, nestingLevel);
1485 this._collapsed = consoleMessage.type === SDK.ConsoleMessage.MessageType.StartGroupCollapsed;
1486 /** @type {?UI.Icon} */
1487 this._expandGroupIcon = null;
1488 }
1489
1490 /**
1491 * @param {boolean} collapsed
1492 */
1493 setCollapsed(collapsed) {
1494 this._collapsed = collapsed;
1495 if (this._expandGroupIcon)
1496 this._expandGroupIcon.setIconType(this._collapsed ? 'smallicon-triangle-right' : 'smallicon-triangle-down');
1497 }
1498
1499 /**
1500 * @return {boolean}
1501 */
1502 collapsed() {
1503 return this._collapsed;
1504 }
1505
1506 /**
1507 * @override
1508 * @return {!Element}
1509 */
1510 toMessageElement() {
1511 if (!this._element) {
1512 super.toMessageElement();
1513 this._expandGroupIcon = UI.Icon.create('', 'expand-group-icon');
1514 if (this._repeatCountElement)
1515 this._repeatCountElement.insertBefore(this._expandGroupIcon, this._repeatCountElement.firstChild);
1516 else
1517 this._element.insertBefore(this._expandGroupIcon, this._contentElement);
1518 this.setCollapsed(this._collapsed);
1519 }
1520 return this._element;
1521 }
1522
1523 /**
1524 * @override
1525 */
1526 _showRepeatCountElement() {
1527 super._showRepeatCountElement();
1528 if (this._repeatCountElement && this._expandGroupIcon)
1529 this._repeatCountElement.insertBefore(this._expandGroupIcon, this._repeatCountElement.firstChild);
1530 }
1531};
1532
1533/**
1534 * @const
1535 * @type {number}
1536 */
1537Console.ConsoleViewMessage.MaxLengthForLinks = 40;
1538
1539Console.ConsoleViewMessage._MaxTokenizableStringLength = 10000;
1540
1541Console.ConsoleViewMessage._LongStringVisibleLength = 5000;