mirror of
https://github.com/flutter/flutter.git
synced 2025-06-03 00:51:18 +00:00
310 lines
8.6 KiB
Dart
310 lines
8.6 KiB
Dart
// Copyright 2014 The Flutter Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
import 'dart:convert';
|
|
import 'dart:io';
|
|
|
|
import 'package:meta/meta.dart';
|
|
import 'package:path/path.dart' as path;
|
|
|
|
import 'constants.dart';
|
|
|
|
/// The location of the Flutter root directory, based on the known location of
|
|
/// this script.
|
|
final Directory flutterRoot = Directory(path.dirname(Platform.script.toFilePath())).parent.parent.parent.parent;
|
|
String get dataRoot => testDataRoot ?? _dataRoot;
|
|
String _dataRoot = path.join(flutterRoot.path, 'dev', 'tools', 'gen_keycodes', 'data');
|
|
|
|
/// Allows overriding of the [dataRoot] for testing purposes.
|
|
@visibleForTesting
|
|
String? testDataRoot;
|
|
|
|
/// Converts `FOO_BAR` to `FooBar`.
|
|
String shoutingToUpperCamel(String shouting) {
|
|
final RegExp initialLetter = RegExp(r'(?:_|^)([^_])([^_]*)');
|
|
final String snake = shouting.toLowerCase();
|
|
final String result = snake.replaceAllMapped(initialLetter, (Match match) {
|
|
return match.group(1)!.toUpperCase() + match.group(2)!.toLowerCase();
|
|
});
|
|
return result;
|
|
}
|
|
|
|
/// Converts 'FooBar' to 'fooBar'.
|
|
///
|
|
/// 'TVFoo' should be convert to 'tvFoo'.
|
|
/// 'KeyX' should be convert to 'keyX'.
|
|
String upperCamelToLowerCamel(String upperCamel) {
|
|
final RegExp initialGroup = RegExp(r'^([A-Z]([A-Z]*|[^A-Z]*))([A-Z]([^A-Z]|$)|$)');
|
|
return upperCamel.replaceFirstMapped(initialGroup, (Match match) {
|
|
return match.group(1)!.toLowerCase() + (match.group(3) ?? '');
|
|
});
|
|
}
|
|
|
|
/// Converts 'fooBar' to 'FooBar'.
|
|
String lowerCamelToUpperCamel(String lowerCamel) {
|
|
return lowerCamel.substring(0, 1).toUpperCase() + lowerCamel.substring(1);
|
|
}
|
|
|
|
/// A list of Dart reserved words.
|
|
///
|
|
/// Since these are Dart reserved words, we can't use them as-is for enum names.
|
|
const List<String> kDartReservedWords = <String>[
|
|
'abstract',
|
|
'as',
|
|
'assert',
|
|
'async',
|
|
'await',
|
|
'break',
|
|
'case',
|
|
'catch',
|
|
'class',
|
|
'const',
|
|
'continue',
|
|
'covariant',
|
|
'default',
|
|
'deferred',
|
|
'do',
|
|
'dynamic',
|
|
'else',
|
|
'enum',
|
|
'export',
|
|
'extends',
|
|
'external',
|
|
'factory',
|
|
'false',
|
|
'final',
|
|
'finally',
|
|
'for',
|
|
'Function',
|
|
'get',
|
|
'hide',
|
|
'if',
|
|
'implements',
|
|
'import',
|
|
'in',
|
|
'interface',
|
|
'is',
|
|
'library',
|
|
'mixin',
|
|
'new',
|
|
'null',
|
|
'on',
|
|
'operator',
|
|
'part',
|
|
'rethrow',
|
|
'return',
|
|
'set',
|
|
'show',
|
|
'static',
|
|
'super',
|
|
'switch',
|
|
'sync',
|
|
'this',
|
|
'throw',
|
|
'true',
|
|
'try',
|
|
'typedef',
|
|
'var',
|
|
'void',
|
|
'while',
|
|
'with',
|
|
'yield',
|
|
];
|
|
|
|
/// Converts an integer into a hex string with the given number of digits.
|
|
String toHex(int? value, {int digits = 8}) {
|
|
if (value == null) {
|
|
return 'null';
|
|
}
|
|
return '0x${value.toRadixString(16).padLeft(digits, '0')}';
|
|
}
|
|
|
|
/// Parses an integer from a hex string.
|
|
int getHex(String input) {
|
|
return int.parse(input, radix: 16);
|
|
}
|
|
|
|
/// Given an [input] string, wraps the text at 80 characters and prepends each
|
|
/// line with the [prefix] string. Use for generated comments.
|
|
String wrapString(String input, {required String prefix}) {
|
|
final int wrapWidth = 80 - prefix.length;
|
|
final StringBuffer result = StringBuffer();
|
|
final List<String> words = input.split(RegExp(r'\s+'));
|
|
String currentLine = words.removeAt(0);
|
|
for (final String word in words) {
|
|
if ((currentLine.length + word.length) < wrapWidth) {
|
|
currentLine += ' $word';
|
|
} else {
|
|
result.writeln('$prefix$currentLine');
|
|
currentLine = word;
|
|
}
|
|
}
|
|
if (currentLine.isNotEmpty) {
|
|
result.writeln('$prefix$currentLine');
|
|
}
|
|
return result.toString();
|
|
}
|
|
|
|
/// Run `fn` with each corresponding element from list1 and list2.
|
|
///
|
|
/// If `list1` has a different length from `list2`, the execution is aborted
|
|
/// after printing an error.
|
|
///
|
|
/// An null list is considered a list with length 0.
|
|
void zipStrict<T1, T2>(Iterable<T1> list1, Iterable<T2> list2, void Function(T1, T2) fn) {
|
|
if (list1 == null && list2 == null)
|
|
return;
|
|
assert(list1.length == list2.length);
|
|
final Iterator<T1> it1 = list1.iterator;
|
|
final Iterator<T2> it2 = list2.iterator;
|
|
while (it1.moveNext()) {
|
|
it2.moveNext();
|
|
fn(it1.current, it2.current);
|
|
}
|
|
}
|
|
|
|
/// Read a Map<String, String> out of its string representation in JSON.
|
|
Map<String, String> parseMapOfString(String jsonString) {
|
|
return (json.decode(jsonString) as Map<String, dynamic>).cast<String, String>();
|
|
}
|
|
|
|
/// Read a Map<String, List<String>> out of its string representation in JSON.
|
|
Map<String, List<String>> parseMapOfListOfString(String jsonString) {
|
|
final Map<String, List<dynamic>> dynamicMap = (json.decode(jsonString) as Map<String, dynamic>).cast<String, List<dynamic>>();
|
|
return dynamicMap.map<String, List<String>>((String key, List<dynamic> value) {
|
|
return MapEntry<String, List<String>>(key, value.cast<String>());
|
|
});
|
|
}
|
|
|
|
Map<String, List<String?>> parseMapOfListOfNullableString(String jsonString) {
|
|
final Map<String, List<dynamic>> dynamicMap = (json.decode(jsonString) as Map<String, dynamic>).cast<String, List<dynamic>>();
|
|
return dynamicMap.map<String, List<String?>>((String key, List<dynamic> value) {
|
|
return MapEntry<String, List<String?>>(key, value.cast<String?>());
|
|
});
|
|
}
|
|
|
|
Map<String, bool> parseMapOfBool(String jsonString) {
|
|
return (json.decode(jsonString) as Map<String, dynamic>).cast<String, bool>();
|
|
}
|
|
|
|
/// Reverse the map of { fromValue -> list of toValue } to { toValue -> fromValue } and return.
|
|
Map<String, String> reverseMapOfListOfString(Map<String, List<String>> inMap, void Function(String fromValue, String newToValue) onDuplicate) {
|
|
final Map<String, String> result = <String, String>{};
|
|
inMap.forEach((String fromValue, List<String> toValues) {
|
|
for (final String toValue in toValues) {
|
|
if (result.containsKey(toValue)) {
|
|
onDuplicate(fromValue, toValue);
|
|
continue;
|
|
}
|
|
result[toValue] = fromValue;
|
|
}
|
|
});
|
|
return result;
|
|
}
|
|
|
|
/// Remove entries whose value `isEmpty` or is null, and return the map.
|
|
///
|
|
/// Will modify the input map.
|
|
Map<String, dynamic> removeEmptyValues(Map<String, dynamic> map) {
|
|
return map..removeWhere((String key, dynamic value) {
|
|
if (value == null)
|
|
return true;
|
|
if (value is Map<String, dynamic>) {
|
|
final Map<String, dynamic> regularizedMap = removeEmptyValues(value);
|
|
return regularizedMap.isEmpty;
|
|
}
|
|
if (value is Iterable<dynamic>) {
|
|
return value.isEmpty;
|
|
}
|
|
return false;
|
|
});
|
|
}
|
|
|
|
void addNameValue(List<String> names, List<int> values, String name, int value) {
|
|
final int foundIndex = values.indexOf(value);
|
|
if (foundIndex == -1) {
|
|
names.add(name);
|
|
values.add(value);
|
|
} else {
|
|
if (!RegExp(r'(^|, )abc1($|, )').hasMatch(name)) {
|
|
names[foundIndex] = '${names[foundIndex]}, $name';
|
|
}
|
|
}
|
|
}
|
|
|
|
enum DeduplicateBehavior {
|
|
// Warn the duplicate entry.
|
|
kWarn,
|
|
|
|
// Skip the latter duplicate entry.
|
|
kSkip,
|
|
|
|
// Keep all duplicate entries.
|
|
kKeep,
|
|
}
|
|
|
|
/// The information for a line used by [OutputLines].
|
|
class OutputLine<T extends Comparable<Object>> {
|
|
OutputLine(this.key, String value)
|
|
: values = <String>[value];
|
|
|
|
final T key;
|
|
final List<String> values;
|
|
}
|
|
|
|
/// A utility class to build join a number of lines in a sorted order.
|
|
///
|
|
/// Use [add] to add a line and associate it with an index. Use [sortedJoin] to
|
|
/// get the joined string of these lines joined sorting them in the order of the
|
|
/// index.
|
|
class OutputLines<T extends Comparable<Object>> {
|
|
OutputLines(this.mapName, {this.behavior = DeduplicateBehavior.kWarn});
|
|
|
|
/// What to do if there are entries with the same key.
|
|
final DeduplicateBehavior behavior;
|
|
|
|
/// The name for this map.
|
|
///
|
|
/// Used in warning messages.
|
|
final String mapName;
|
|
|
|
final Map<T, OutputLine<T>> lines = <T, OutputLine<T>>{};
|
|
|
|
void add(T key, String line) {
|
|
final OutputLine<T>? existing = lines[key];
|
|
if (existing != null) {
|
|
switch (behavior) {
|
|
case DeduplicateBehavior.kWarn:
|
|
print('Warn: Request to add $key to map "$mapName" as:\n $line\n but it already exists as:\n ${existing.values[0]}');
|
|
return;
|
|
case DeduplicateBehavior.kSkip:
|
|
return;
|
|
case DeduplicateBehavior.kKeep:
|
|
existing.values.add(line);
|
|
return;
|
|
}
|
|
}
|
|
lines[key] = OutputLine<T>(key, line);
|
|
}
|
|
|
|
String join() {
|
|
return lines.values.map((OutputLine<T> line) => line.values.join('\n')).join('\n');
|
|
}
|
|
|
|
String sortedJoin() {
|
|
return (lines.values.toList()
|
|
..sort((OutputLine<T> a, OutputLine<T> b) => a.key.compareTo(b.key)))
|
|
.map((OutputLine<T> line) => line.values.join('\n'))
|
|
.join('\n');
|
|
}
|
|
}
|
|
|
|
int toPlane(int value, int plane) {
|
|
return (value & kValueMask.value) + (plane & kPlaneMask.value);
|
|
}
|
|
|
|
int getPlane(int value) {
|
|
return value & kPlaneMask.value;
|
|
}
|