// Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file // for details. All rights reserved. Use of this source code is governed by a // BSD-style license that can be found in the LICENSE file. import 'description.dart'; import 'interfaces.dart'; import 'util.dart'; /// Returns a pretty-printed representation of [object]. /// /// If [maxLineLength] is passed, this will attempt to ensure that each line is /// no longer than [maxLineLength] characters long. This isn't guaranteed, since /// individual objects may have string representations that are too long, but /// most lines will be less than [maxLineLength] long. /// /// If [maxItems] is passed, [Iterable]s and [Map]s will only print their first /// [maxItems] members or key/value pairs, respectively. String prettyPrint(Object? object, {int? maxLineLength, int? maxItems}) { String prettyPrintImpl( Object? object, int indent, Set seen, bool top, ) { // If the object is a matcher, use its description. if (object is Matcher) { var description = StringDescription(); object.describe(description); return '<$description>'; } // Avoid looping infinitely on recursively-nested data structures. if (seen.contains(object)) return '(recursive)'; seen = seen.union({object}); String pp(Object? child) => prettyPrintImpl(child, indent + 2, seen, false); if (object is Iterable) { // Print the type name for non-List iterables. var type = object is List ? '' : '${_typeName(object)}:'; // Truncate the list of strings if it's longer than [maxItems]. var strings = []; for (var item in object) { if (maxItems != null && strings.length >= maxItems) { strings[maxItems - 1] = '...'; break; } strings.add(pp(item)); } // If the printed string is short and doesn't contain a newline, print it // as a single line. var singleLine = "$type[${strings.join(', ')}]"; if ((maxLineLength == null || singleLine.length + indent <= maxLineLength) && !singleLine.contains('\n')) { return singleLine; } // Otherwise, print each member on its own line. return '$type[\n${strings.map((string) { return _indent(indent + 2) + string; }).join(',\n')}\n${_indent(indent)}]'; } else if (object is Map) { // Truncate the list of strings if it's longer than [maxItems]. var strings = []; for (var key in object.keys) { if (maxItems != null && strings.length >= maxItems) { strings[maxItems - 1] = '...'; break; } strings.add('${pp(key)}: ${pp(object[key])}'); } // If the printed string is short and doesn't contain a newline, print it // as a single line. var singleLine = '{${strings.join(", ")}}'; if ((maxLineLength == null || singleLine.length + indent <= maxLineLength) && !singleLine.contains('\n')) { return singleLine; } // Otherwise, print each key/value pair on its own line. return '{\n${strings.map((string) { return _indent(indent + 2) + string; }).join(',\n')}\n${_indent(indent)}}'; } else if (object is String) { // Escape strings and print each line on its own line. var value = object .split('\n') .map(_escapeString) .join("\\n'\n${_indent(indent + 2)}'"); return "'$value'"; } else { var value = object.toString().replaceAll('\n', '${_indent(indent)}\n'); var defaultToString = value.startsWith('Instance of '); // If this is the top-level call to [prettyPrint], wrap the value on angle // brackets to set it apart visually. if (top) value = '<$value>'; // Print the type of objects with custom [toString] methods. Primitive // objects and objects that don't implement a custom [toString] don't need // to have their types printed. if (object is num || object is bool || object is Function || object is RegExp || object is MapEntry || object is Expando || object == null || defaultToString) { return value; } else { return '${_typeName(object)}:$value'; } } } return prettyPrintImpl(object, 0, {}, true); } String _indent(int length) => List.filled(length, ' ').join(''); /// Returns the name of the type of [x] with fallbacks for core types with /// private implementations. String _typeName(Object x) { if (x is Type) return 'Type'; if (x is Uri) return 'Uri'; if (x is Set) return 'Set'; if (x is BigInt) return 'BigInt'; return '${x.runtimeType}'; } /// Returns [source] with any control characters replaced by their escape /// sequences. /// /// This doesn't add quotes to the string, but it does escape single quote /// characters so that single quotes can be applied externally. String _escapeString(String source) => escape(source).replaceAll("'", r"\'");