dart-sass/lib/src/utils.dart

272 lines
9.1 KiB
Dart
Raw Normal View History

2016-05-24 03:01:15 +02:00
// 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.
2016-05-24 23:01:41 +02:00
import 'dart:collection';
2016-08-05 02:25:57 +02:00
import 'dart:math' as math;
2016-05-24 23:01:41 +02:00
import 'package:charcode/charcode.dart';
2016-08-13 00:24:27 +02:00
import 'package:collection/collection.dart';
2016-05-24 03:01:15 +02:00
import 'package:source_span/source_span.dart';
import 'ast/node.dart';
import 'io.dart';
2016-09-22 01:04:15 +02:00
import 'util/character.dart';
2016-05-24 03:01:15 +02:00
2016-10-15 11:57:29 +02:00
/// Converts [iter] into a sentence, separating each word with [conjunction].
2016-08-27 10:46:09 +02:00
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';
}
2016-10-15 11:57:29 +02:00
/// Flattens the first level of nested arrays in [iterable].
///
/// The return value is ordered first by index in the nested iterable, then by
/// the index *of* that iterable in [iterable]. For example,
/// `flattenVertically([["1a", "1b"], ["2a", "2b"]])` returns `["1a", "2a",
/// "1b", "2b"]`.
2017-05-19 01:47:22 +02:00
List<T> flattenVertically<T>(Iterable<Iterable<T>> iterable) {
2016-09-24 14:40:01 +02:00
var queues = iterable.map((inner) => new QueueList.from(inner)).toList();
if (queues.length == 1) return queues.first;
2017-05-19 01:47:22 +02:00
var result = <T>[];
2016-09-24 14:40:01 +02:00
while (queues.isNotEmpty) {
queues.removeWhere((queue) {
result.add(queue.removeFirst());
return queue.isEmpty;
});
}
return result;
}
2016-10-15 11:57:29 +02:00
/// Converts [codepointIndex] to a code unit index, relative to [string].
///
/// A codepoint index is the index in pure Unicode codepoints; a code unit index
/// is an index into a UTF-16 string.
2016-09-22 00:53:54 +02:00
int codepointIndexToCodeUnitIndex(String string, int codepointIndex) {
var codeUnitIndex = 0;
for (var i = 0; i < codepointIndex; i++) {
2016-09-22 01:04:15 +02:00
if (isHighSurrogate(string.codeUnitAt(codeUnitIndex++))) codeUnitIndex++;
2016-09-22 00:53:54 +02:00
}
return codeUnitIndex;
}
2016-10-15 11:57:29 +02:00
/// Converts [codeUnitIndex] to a codepoint index, relative to [string].
///
/// A codepoint index is the index in pure Unicode codepoints; a code unit index
/// is an index into a UTF-16 string.
2016-09-22 01:04:15 +02:00
int codeUnitIndexToCodepointIndex(String string, int codeUnitIndex) {
var codepointIndex = 0;
for (var i = 0; i < codeUnitIndex; i++) {
codepointIndex++;
2016-10-19 05:55:16 +02:00
if (isHighSurrogate(string.codeUnitAt(i))) i++;
2016-09-22 01:04:15 +02:00
}
return codepointIndex;
}
2016-10-15 11:57:29 +02:00
/// Returns whether [list1] and [list2] have the same contents.
2017-05-28 02:18:13 +02:00
bool listEquals<T>(List<T> list1, List<T> list2) =>
const ListEquality().equals(list1, list2);
2016-08-13 00:24:27 +02:00
2016-10-15 11:57:29 +02:00
/// Returns a hash code for [list] that matches [listEquals].
2016-08-13 01:53:19 +02:00
int listHash(List list) => const ListEquality().hash(list);
2016-10-15 11:57:29 +02:00
/// Returns a source span that covers the spans of both the first and last nodes
/// in [nodes].
///
/// If [nodes] is empty, or if either the first or last node has a `null` span,
/// returns `null`.
2016-06-06 09:05:04 +02:00
FileSpan spanForList(List<AstNode> nodes) {
2016-05-24 03:01:15 +02:00
if (nodes.isEmpty) return null;
2016-09-23 01:23:51 +02:00
// Spans may be null for dynamically-constructed ASTs.
if (nodes.first.span == null) return null;
if (nodes.last.span == null) return null;
2016-06-06 09:05:04 +02:00
return nodes.first.span.expand(nodes.last.span);
2016-05-24 03:01:15 +02:00
}
2016-05-24 23:01:41 +02:00
2016-10-15 11:57:29 +02:00
/// Returns [name] without a vendor prefix.
///
/// If [name] has no vendor prefix, it's returned as-is.
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;
}
2016-10-15 11:57:29 +02:00
/// Returns whether [string1] and [string2] are equal if `-` and `_` are
/// considered equivalent.
2016-08-20 00:54:57 +02:00
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;
}
2016-10-15 11:57:29 +02:00
/// Returns a hash code for [string] that matches [equalsIgnoreSeparator].
2016-08-20 00:54:57 +02:00
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;
}
2016-10-15 11:57:29 +02:00
/// Returns whether [string1] and [string2] are equal, ignoring ASCII case.
bool equalsIgnoreCase(String string1, String string2) {
2016-08-20 00:54:57 +02:00
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();
2016-05-24 23:01:41 +02:00
}
2016-06-03 22:36:20 +02:00
2016-10-15 11:57:29 +02:00
/// Returns an empty map that uses [equalsIgnoreSeparator] for key equality.
2016-10-20 02:56:48 +02:00
///
/// If [source] is passed, copies it into the map.
2017-05-19 01:47:22 +02:00
Map<String, V> normalizedMap<V>([Map<String, V> source]) {
var map = new LinkedHashMap<String, V>(
2016-10-20 02:56:48 +02:00
equals: equalsIgnoreSeparator, hashCode: hashCodeIgnoreSeparator);
if (source != null) map.addAll(source);
return map;
}
2016-08-27 10:46:09 +02:00
2016-10-15 11:57:29 +02:00
/// Returns an empty set that uses [equalsIgnoreSeparator] for equality.
2016-10-20 02:56:48 +02:00
///
/// If [source] is passed, copies it into the set.
Set<String> normalizedSet([Iterable<String> source]) {
var set = new LinkedHashSet(
equals: equalsIgnoreSeparator, hashCode: hashCodeIgnoreSeparator);
if (source != null) set.addAll(source);
return set;
}
2016-08-20 00:54:57 +02:00
2016-10-15 11:57:29 +02:00
/// Like [mapMap], but returns a map that uses [equalsIgnoreSeparator] for key
/// equality.
2017-05-19 01:47:22 +02:00
Map<String, V2> normalizedMapMap<K, V1, V2>(Map<K, V1> map,
{String key(K key, V1 value), V2 value(K key, V1 value)}) {
key ??= (mapKey, _) => mapKey as String;
2017-05-19 01:47:22 +02:00
value ??= (_, mapValue) => mapValue as V2;
2017-05-19 01:47:22 +02:00
var result = normalizedMap<V2>();
map.forEach((mapKey, mapValue) {
result[key(mapKey, mapValue)] = value(mapKey, mapValue);
});
return result;
}
2016-10-15 11:57:29 +02:00
/// Returns the longest common subsequence between [list1] and [list2].
///
/// If there are more than one equally long common subsequence, returns the one
/// which starts first in [list1].
///
/// If [select] is passed, it's used to check equality between elements in each
/// list. If it returns `null`, the elements are considered unequal; otherwise,
/// it should return the element to include in the return value.
2017-05-19 01:47:22 +02:00
List<T> longestCommonSubsequence<T>(List<T> list1, List<T> list2,
{T select(T element1, T element2)}) {
2016-08-02 20:36:34 +02:00
select ??= (element1, element2) => element1 == element2 ? element1 : null;
var lengths = new List.generate(
list1.length + 1, (_) => new List.filled(list2.length + 1, 0),
growable: false);
2017-05-19 01:47:22 +02:00
var selections = new List<List<T>>.generate(
list1.length, (_) => new List<T>(list2.length),
2016-08-02 20:36:34 +02:00
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;
}
}
2017-05-19 01:47:22 +02:00
List<T> backtrack(int i, int j) {
2016-08-02 20:36:34 +02:00
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);
}
2016-09-20 02:20:35 +02:00
2016-10-15 11:57:29 +02:00
/// Removes and returns the first value in [list] that matches [test].
///
/// By default, throws a [StateError] if no value matches. If [orElse] is
/// passed, its return value is used instead.
2017-05-19 01:47:22 +02:00
T removeFirstWhere<T>(List<T> list, bool test(T value), {T orElse()}) {
T toRemove;
2016-09-20 02:20:35 +02:00
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;
}
}
/// Rotates the element in list from [start] (inclusive) to [end] (exclusive)
/// one index higher, looping the final element back to [start].
void rotateSlice(List list, int start, int end) {
var element = list[end - 1];
for (var i = start; i < end; i++) {
var next = list[i];
list[i] = element;
element = next;
}
}
/// Prints a warning to standard error, associated with [span].
///
/// If [color] is `true`, this uses terminal colors.
void warn(String message, FileSpan span, {bool color: false}) {
var warning = color ? '\u001b[33m\u001b[1mWarning\u001b[0m' : 'WARNING';
stderr.writeln("$warning on ${span.message("\n$message", color: color)}\n");
}