// Copyright (c) 2014, 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 'charcode.dart'; import 'location.dart'; import 'span.dart'; import 'span_with_context.dart'; /// Returns the minimum of [obj1] and [obj2] according to /// [Comparable.compareTo]. T min>(T obj1, T obj2) => obj1.compareTo(obj2) > 0 ? obj2 : obj1; /// Returns the maximum of [obj1] and [obj2] according to /// [Comparable.compareTo]. T max>(T obj1, T obj2) => obj1.compareTo(obj2) > 0 ? obj1 : obj2; /// Returns whether all elements of [iter] are the same value, according to /// `==`. bool isAllTheSame(Iterable iter) { if (iter.isEmpty) return true; final firstValue = iter.first; for (var value in iter.skip(1)) { if (value != firstValue) { return false; } } return true; } /// Returns whether [span] covers multiple lines. bool isMultiline(SourceSpan span) => span.start.line != span.end.line; /// Sets the first `null` element of [list] to [element]. void replaceFirstNull(List list, E element) { final index = list.indexOf(null); if (index < 0) throw ArgumentError('$list contains no null elements.'); list[index] = element; } /// Sets the element of [list] that currently contains [element] to `null`. void replaceWithNull(List list, E element) { final index = list.indexOf(element); if (index < 0) { throw ArgumentError('$list contains no elements matching $element.'); } list[index] = null; } /// Returns the number of instances of [codeUnit] in [string]. int countCodeUnits(String string, int codeUnit) { var count = 0; for (var codeUnitToCheck in string.codeUnits) { if (codeUnitToCheck == codeUnit) count++; } return count; } /// Finds a line in [context] containing [text] at the specified [column]. /// /// Returns the index in [context] where that line begins, or null if none /// exists. int? findLineStart(String context, String text, int column) { // If the text is empty, we just want to find the first line that has at least // [column] characters. if (text.isEmpty) { var beginningOfLine = 0; while (true) { final index = context.indexOf('\n', beginningOfLine); if (index == -1) { return context.length - beginningOfLine >= column ? beginningOfLine : null; } if (index - beginningOfLine >= column) return beginningOfLine; beginningOfLine = index + 1; } } var index = context.indexOf(text); while (index != -1) { // Start looking before [index] in case [text] starts with a newline. final lineStart = index == 0 ? 0 : context.lastIndexOf('\n', index - 1) + 1; final textColumn = index - lineStart; if (column == textColumn) return lineStart; index = context.indexOf(text, index + 1); } // ignore: avoid_returning_null return null; } /// Returns a two-element list containing the start and end locations of the /// span from [start] code units (inclusive) to [end] code units (exclusive) /// after the beginning of [span]. /// /// This is factored out so it can be shared between /// [SourceSpanExtension.subspan] and [SourceSpanWithContextExtension.subspan]. List subspanLocations(SourceSpan span, int start, [int? end]) { final text = span.text; final startLocation = span.start; var line = startLocation.line; var column = startLocation.column; // Adjust [line] and [column] as necessary if the character at [i] in [text] // is a newline. void consumeCodePoint(int i) { final codeUnit = text.codeUnitAt(i); if (codeUnit == $lf || // A carriage return counts as a newline, but only if it's not // followed by a line feed. (codeUnit == $cr && (i + 1 == text.length || text.codeUnitAt(i + 1) != $lf))) { line += 1; column = 0; } else { column += 1; } } for (var i = 0; i < start; i++) { consumeCodePoint(i); } final newStartLocation = SourceLocation(startLocation.offset + start, sourceUrl: span.sourceUrl, line: line, column: column); SourceLocation newEndLocation; if (end == null || end == span.length) { newEndLocation = span.end; } else if (end == start) { newEndLocation = newStartLocation; } else { for (var i = start; i < end; i++) { consumeCodePoint(i); } newEndLocation = SourceLocation(startLocation.offset + end, sourceUrl: span.sourceUrl, line: line, column: column); } return [newStartLocation, newEndLocation]; }