// Copyright 2016 Google Inc. Use of this source code is governed by an // MIT-style license that can be found in the LICENSE file or at // https://opensource.org/licenses/MIT. import 'dart:collection'; import 'dart:math' as math; import 'package:charcode/charcode.dart'; import 'package:collection/collection.dart'; import 'package:source_span/source_span.dart'; import 'ast/node.dart'; import 'util/character.dart'; import 'value.dart'; const _epsilon = 1 / (10 * SassNumber.precision); class LinkedListValue extends LinkedListEntry> { final T value; LinkedListValue(this.value); } String toSentence(Iterable iter, [String conjunction]) { conjunction ??= "and"; if (iter.length == 1) return iter.first.toString(); return iter.take(iter.length - 1).join(", ") + " $conjunction ${iter.last}"; } /// Returns [name] if [number] is 1, or the plural of [name] otherwise. /// /// By default, this just adds "s" to the end of [name] to get the plural. If /// [plural] is passed, that's used instead. String pluralize(String name, int number, {String plural}) { if (number == 1) return name; if (plural != null) return plural; return '${name}s'; } int codepointIndexToCodeUnitIndex(String string, int codepointIndex) { var codeUnitIndex = 0; for (var i = 0; i < codepointIndex; i++) { if (isHighSurrogate(string.codeUnitAt(codeUnitIndex++))) codeUnitIndex++; } return codeUnitIndex; } int codeUnitIndexToCodepointIndex(String string, int codeUnitIndex) { var codepointIndex = 0; for (var i = 0; i < codeUnitIndex; i++) { codepointIndex++; if (isHighSurrogate(string.codeUnitAt(i++))) i++; } return codepointIndex; } bool listEquals/**/(List/**/ list1, List/**/ list2) => const ListEquality().equals(list1, list2); int listHash(List list) => const ListEquality().hash(list); FileSpan spanForList(List nodes) { if (nodes.isEmpty) return null; // Spans may be null for dynamically-constructed ASTs. if (nodes.first.span == null) return null; if (nodes.last.span == null) return null; return nodes.first.span.expand(nodes.last.span); } String unvendor(String name) { if (name.length < 2) return name; if (name.codeUnitAt(0) != $dash) return name; if (name.codeUnitAt(1) == $dash) return name; for (var i = 2; i < name.length; i++) { if (name.codeUnitAt(i) == $dash) return name.substring(i + 1); } return name; } bool equalsIgnoreSeparator(String string1, String string2) { if (identical(string1, string2)) return true; if (string1 == null || string2 == null) return false; if (string1.length != string2.length) return false; for (var i = 0; i < string1.length; i++) { var codeUnit1 = string1.codeUnitAt(i); var codeUnit2 = string2.codeUnitAt(i); if (codeUnit1 == codeUnit2) continue; if (codeUnit1 == $dash) { if (codeUnit2 != $underscore) return false; } else if (codeUnit1 == $underscore) { if (codeUnit2 != $dash) return false; } else { return false; } } return true; } int hashCodeIgnoreSeparator(String string) { var hash = 4603; for (var i = 0; i < string.length; i++) { var codeUnit = string.codeUnitAt(i); if (codeUnit == $underscore) codeUnit = $dash; hash &= 0x3FFFFFF; hash *= 33; hash ^= codeUnit; } return hash; } bool equalsIgnoreCase(String string1, String string2) { if (identical(string1, string2)) return true; if (string1 == null || string2 == null) return false; if (string1.length != string2.length) return false; return string1.toUpperCase() == string2.toUpperCase(); } Map/**/ normalizedMap/**/() => new LinkedHashMap( equals: equalsIgnoreSeparator, hashCode: hashCodeIgnoreSeparator); Set normalizedSet() => new LinkedHashSet( equals: equalsIgnoreSeparator, hashCode: hashCodeIgnoreSeparator); Map/**/ normalizedMapMap/**/(Map/**/ map, {String key(/*=K*/ key, /*=V1*/ value), /*=V2*/ value(/*=K*/ key, /*=V1*/ value)}) { key ??= (mapKey, _) => mapKey as String; value ??= (_, mapValue) => mapValue as dynamic/*=V2*/; var result = normalizedMap/**/(); map.forEach((mapKey, mapValue) { result[key(mapKey, mapValue)] = value(mapKey, mapValue); }); return result; } bool fuzzyEquals(num number1, num number2) => (number1 - number2).abs() < _epsilon; int fuzzyHashCode(num number) => (number % _epsilon).hashCode; bool fuzzyLessThan(num number1, num number2) => number1 < number2 && !fuzzyEquals(number1, number2); bool fuzzyLessThanOrEquals(num number1, num number2) => number1 < number2 || fuzzyEquals(number1, number2); bool fuzzyGreaterThan(num number1, num number2) => number1 > number2 && !fuzzyEquals(number1, number2); bool fuzzyGreaterThanOrEquals(num number1, num number2) => number1 > number2 || fuzzyEquals(number1, number2); int fuzzyRound(num number) { // If the number is within epsilon of X.5, round up (or down for negative // numbers). if (fuzzyLessThan(number % 1, 0.5)) return number.round(); return number > 0 ? number.ceil() : number.floor(); } num fuzzyCheckRange(num number, num min, num max) { if (fuzzyEquals(number, min)) return min; if (fuzzyEquals(number, max)) return max; if (number > min && number < max) return number; return null; } num fuzzyAssertRange(num number, num min, num max, [String name]) { var result = fuzzyCheckRange(number, min, max); if (result != null) return result; throw new RangeError.value(number, name, "must be between $min and $max."); } List/**/ longestCommonSubsequence/**/( List/**/ list1, List/**/ list2, {/*=T*/ select(/*=T*/ element1, /*=T*/ element2)}) { select ??= (element1, element2) => element1 == element2 ? element1 : null; var lengths = new List.generate( list1.length + 1, (_) => new List.filled(list2.length + 1, 0), growable: false); var selections = new List*/ >.generate( list1.length, (_) => new List/**/(list2.length), growable: false); // TODO(nweiz): Calling [select] here may be expensive. Can we use a memoizing // approach to avoid calling it O(n*m) times in most cases? for (var i = 0; i < list1.length; i++) { for (var j = 0; j < list2.length; j++) { var selection = select(list1[i], list2[j]); selections[i][j] = selection; lengths[i + 1][j + 1] = selection == null ? math.max(lengths[i + 1][j], lengths[i][j + 1]) : lengths[i][j] + 1; } } List/**/ backtrack(int i, int j) { if (i == -1 || j == -1) return []; var selection = selections[i][j]; if (selection != null) return backtrack(i - 1, j - 1)..add(selection); return lengths[i + 1][j] > lengths[i][j + 1] ? backtrack(i, j - 1) : backtrack(i - 1, j); } return backtrack(list1.length - 1, list2.length - 1); } /*=T*/ removeFirstWhere/**/(List/**/ list, bool test(/*=T*/ value), {/*=T*/ orElse()}) { var/*=T*/ toRemove; for (var element in list) { if (!test(element)) continue; toRemove = element; break; } if (toRemove == null) { if (orElse != null) return orElse(); throw new StateError("No such element."); } else { list.remove(toRemove); return toRemove; } }