blob: c816ea1d5c22675e5a073c86ba56ef0ed2d72885 [file] [log] [blame]
// Copyright (c) 2023, 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.
// Check that JSStringImpl behaves correctly as a subtype of String.
import 'dart:js_interop';
import 'package:expect/expect.dart';
// We run many tests in three configurations:
// 1) Test should ensure receivers for all [String] operations will be
// `JSStringImpl`.
// 2) Test should ensure arguments to all [String] operations
// will be `JSStringImpl`.
// 3) Test should ensure both receivers and arguments
// for all [String] operations will be `JSStringImpl`.
enum TestMode {
jsStringImplReceiver,
jsStringImplArgument,
jsStringImplReceiverAndArguments,
}
enum Position { jsStringImplReceiver, jsStringImplArgument }
bool useJSStringImpl(Position pos, TestMode mode) =>
(pos == Position.jsStringImplReceiver &&
(mode == TestMode.jsStringImplReceiver ||
mode == TestMode.jsStringImplReceiverAndArguments)) ||
(pos == Position.jsStringImplArgument &&
(mode == TestMode.jsStringImplArgument ||
mode == TestMode.jsStringImplReceiverAndArguments));
// Round trip through `dart:js_interop` to get a `JSString` backed `String`.
String jsStringImpl(String s) => s.toJS.toDart;
String getStr(String s, Position pos, TestMode mode) =>
useJSStringImpl(pos, mode) ? jsStringImpl(s) : s;
void testStringLength(int length, String str) {
Expect.equals(length, str.length);
(length == 0 ? Expect.isTrue : Expect.isFalse)(str.isEmpty);
(length != 0 ? Expect.isTrue : Expect.isFalse)(str.isNotEmpty);
}
void testLength(TestMode mode) {
String str = getStr('', Position.jsStringImplReceiver, mode);
for (var i = 0; i < 5; i++) {
testStringLength(i, str);
str += getStr(' ', Position.jsStringImplArgument, mode);
}
}
void testOutOfRange() {
String a = jsStringImpl('Hello');
bool exception_caught = false;
try {
var c = a[20]; // Throw exception.
} on RangeError catch (e) {
exception_caught = true;
}
Expect.isTrue(exception_caught);
}
void testIndex() {
String str = jsStringImpl('string');
List<String> chars = ['s', 't', 'r', 'i', 'n', 'g'];
assert(str.length == chars.length);
for (int i = 0; i < str.length; i++) {
Expect.isTrue(str[i] is String);
testStringLength(1, str[i]);
Expect.equals(str[i], chars[i]);
}
}
void testCodeUnitAt() {
String str = jsStringImpl('string');
List<int> codes = [115, 116, 114, 105, 110, 103];
for (int i = 0; i < str.length; i++) {
Expect.isTrue(str.codeUnitAt(i) is int);
Expect.equals(str.codeUnitAt(i), codes[i]);
}
}
void testConcat(TestMode mode) {
var a = getStr('One', Position.jsStringImplReceiver, mode);
var b = getStr('Four', Position.jsStringImplArgument, mode);
var c = a + b;
testStringLength(7, c);
Expect.equals('OneFour', c);
}
void testEquals(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
final rstr = r('str');
final astr = a('str');
Expect.equals(rstr, astr);
Expect.equals(rstr, a('s') + 't' + 'r');
Expect.equals(r('s') + 't' + 'r', a('str'));
Expect.isFalse(rstr == a('s'));
Expect.isFalse(r('s') == astr);
Expect.isFalse(r('') == a('s'));
Expect.equals(r(''), a(''));
}
void testEndsWith(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
final rstr = r('str');
Expect.isTrue(rstr.endsWith(a('str')));
Expect.isFalse(rstr.endsWith(a('t')));
final rEmptyStr = r('');
Expect.isTrue(rEmptyStr.endsWith(r('')));
Expect.isFalse(rEmptyStr.endsWith(r('s')));
}
void testStartsWith(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
final rstr = r('str');
Expect.isTrue(rstr.startsWith(a('str')));
Expect.isFalse(rstr.startsWith(a('stri')));
final rEmptyStr = r('');
Expect.isTrue(rEmptyStr.startsWith(a('')));
Expect.isFalse(rEmptyStr.startsWith(a('s')));
final rstrstr = r('strstr');
final as = a('s');
Expect.isFalse(rstrstr.startsWith(as, 1));
Expect.isTrue(rstrstr.startsWith(as, 3));
final ast = a('st');
Expect.isFalse(rstrstr.startsWith(ast, 1));
Expect.isTrue(rstrstr.startsWith(ast, 3));
final astr = a('str');
Expect.isFalse(rstrstr.startsWith(astr, 1));
Expect.isTrue(rstrstr.startsWith(astr, 3));
final aEmptyStr = a('');
Expect.isTrue(rstr.startsWith(aEmptyStr, 0));
Expect.throws(() => rstr.startsWith(aEmptyStr, -1));
Expect.throws(() => rstr.startsWith(aEmptyStr, 4));
final regexp = RegExp('s(?:tr?)?');
Expect.isTrue(rstr.startsWith(regexp));
Expect.isFalse(rstrstr.startsWith(regexp, 1));
Expect.isFalse(rstrstr.startsWith(regexp, 2));
Expect.isTrue(rstrstr.startsWith(regexp, 3));
Expect.isFalse(rstrstr.startsWith(regexp, 4));
}
void testIndexOf(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
final rstr = r('str');
final rEmptyStr = r('');
final aEmptyStr = a('');
Expect.equals(0, rstr.indexOf(aEmptyStr, 0));
Expect.equals(0, rEmptyStr.indexOf(aEmptyStr, 0));
Expect.equals(-1, rEmptyStr.indexOf(a('a'), 0));
Expect.equals(1, rstr.indexOf(a('t'), 0));
Expect.equals(-1, rstr.indexOf(a('string'), 0));
final rstrstr = r('strstr');
final astr = a('str');
Expect.equals(0, rstrstr.indexOf(astr, 0));
Expect.equals(3, rstrstr.indexOf(astr, 1));
Expect.equals(3, rstrstr.indexOf(astr, 2));
Expect.equals(3, rstrstr.indexOf(astr, 3));
for (int i = 0; i < 5; i++) {
if (i > rstr.length) {
Expect.throws(() => rstr.indexOf(aEmptyStr, i));
} else {
int result = rstr.indexOf(aEmptyStr, i);
Expect.equals(i, result);
}
}
final banana = a('banana');
var re = RegExp('an?');
Expect.equals(1, banana.indexOf(re));
Expect.equals(1, banana.indexOf(re, 0));
Expect.equals(1, banana.indexOf(re, 1));
Expect.equals(3, banana.indexOf(re, 2));
Expect.equals(3, banana.indexOf(re, 3));
Expect.equals(5, banana.indexOf(re, 4));
Expect.equals(5, banana.indexOf(re, 5));
Expect.equals(-1, banana.indexOf(re, 6));
Expect.throws(() => banana.indexOf(re, -1));
Expect.throws(() => banana.indexOf(re, 7));
re = RegExp('x?');
for (int i = 0; i <= rstr.length; i++) {
Expect.equals(i, rstr.indexOf(re, i));
}
}
void testLastIndexOf(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
final rstr = r('str');
final rEmptyStr = r('');
final aEmptyStr = a('');
Expect.equals(2, rstr.lastIndexOf(aEmptyStr, 2));
Expect.equals(0, rEmptyStr.lastIndexOf(aEmptyStr, 0));
Expect.equals(-1, rEmptyStr.lastIndexOf(a('a'), 0));
final astr = a('str');
final astring = a('string');
final at = a('t');
Expect.equals(1, rstr.lastIndexOf(at));
Expect.equals(0, rstr.lastIndexOf(astr, 2));
Expect.equals(2, rstr.lastIndexOf(a('r'), 2));
Expect.equals(-1, rstr.lastIndexOf(astring, 2));
final rstrstr = r('strstr');
Expect.equals(4, rstrstr.lastIndexOf(at, 5));
Expect.equals(3, rstrstr.lastIndexOf(astr, 5));
Expect.throws(() {
rstr.lastIndexOf(astring, 5);
});
Expect.equals(4, rstrstr.lastIndexOf(at, 5));
Expect.equals(3, rstrstr.lastIndexOf(astr, 5));
Expect.equals(5, rstrstr.lastIndexOf(a('r')));
Expect.equals(5, rstrstr.lastIndexOf(a('r'), null));
for (int i = 0; i < 5; i++) {
if (i > rstr.length) {
Expect.throws(() => rstr.indexOf(aEmptyStr, i));
} else {
int result = rstr.lastIndexOf(aEmptyStr, i);
Expect.equals(i, result);
}
}
final banana = r('banana');
var re = RegExp('an?');
Expect.equals(5, banana.lastIndexOf(re));
Expect.equals(5, banana.lastIndexOf(re, 6));
Expect.equals(5, banana.lastIndexOf(re, 5));
Expect.equals(3, banana.lastIndexOf(re, 4));
Expect.equals(3, banana.lastIndexOf(re, 3));
Expect.equals(1, banana.lastIndexOf(re, 2));
Expect.equals(1, banana.lastIndexOf(re, 1));
Expect.equals(-1, banana.lastIndexOf(re, 0));
Expect.throws(() => banana.lastIndexOf(re, -1));
Expect.throws(() => banana.lastIndexOf(re, 7));
re = RegExp('x?');
for (int i = 0; i <= rstr.length; i++) {
Expect.equals(i, rstr.indexOf(re, i));
}
}
void testHashCode() {
for (final str in ['', 'foobar', 'hello world !']) {
String a = str;
String b = jsStringImpl(str);
Expect.equals(a, b);
Expect.equals(a.hashCode, b.hashCode);
}
}
void testContains(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
final rstr = r('str');
Expect.isTrue(rstr.contains(a('s'), 0));
Expect.isTrue(rstr.contains(a('st'), 0));
Expect.isTrue(rstr.contains(a('str'), 0));
Expect.isTrue(rstr.contains(a('t'), 0));
Expect.isTrue(rstr.contains(a('r'), 0));
Expect.isTrue(rstr.contains(a('tr'), 0));
Expect.isFalse(rstr.contains(a('sr'), 0));
Expect.isFalse(rstr.contains(a('string'), 0));
Expect.isTrue(rstr.contains(a(''), 0));
Expect.isTrue(r('').contains(a(''), 0));
Expect.isFalse(r('').contains(a('s'), 0));
}
void testReplaceAll(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
final aFrom = a('from');
final aTo = a('to');
final aEmptyStr = a('');
Expect.equals('AtoBtoCDtoE', r('AfromBfromCDfromE').replaceAll(aFrom, aTo));
Expect.equals('toABtoCDtoE', r('fromABfromCDfromE').replaceAll(aFrom, aTo));
Expect.equals(
'toABtoCDtoEto',
r('fromABfromCDfromEfrom').replaceAll(aFrom, aTo),
);
Expect.equals('ABC', r('ABC').replaceAll(aFrom, aTo));
Expect.equals('', r('').replaceAll(aFrom, aTo));
Expect.equals('fro', r('fro').replaceAll(aFrom, aTo));
Expect.equals('to', r('from').replaceAll(aFrom, aTo));
Expect.equals('to', r('to').replaceAll(aFrom, aTo));
Expect.equals('', r('from').replaceAll(aFrom, aEmptyStr));
Expect.equals('AB', r('AfromB').replaceAll(aFrom, aEmptyStr));
Expect.equals('to', r('').replaceAll(aEmptyStr, aTo));
Expect.equals('toAtoBtoCto', r('ABC').replaceAll(aEmptyStr, aTo));
Expect.equals('aXXcaXXdae', r('abcabdae').replaceAll(RegExp('b'), a('XX')));
Expect.equals(
'aXXcaXXdae',
r('abcabdae').replaceAll(RegExpWrap('b'), a('XX')),
);
}
void testReplaceAllMapped(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
String mark(Match m) => a('[${m[0]}]');
Expect.equals('a[b]ca[b]dae', r('abcabdae').replaceAllMapped(a('b'), mark));
Expect.equals('abcabdae', r('abcabdae').replaceAllMapped(a('f'), mark));
Expect.equals('', r('').replaceAllMapped(a('from'), mark));
Expect.equals('bcbde', r('abcabdae').replaceAllMapped(a('a'), (m) => a('')));
Expect.equals('[]', r('').replaceAllMapped(a(''), mark));
Expect.equals('[]A[]B[]C[]', r('ABC').replaceAllMapped(a(''), mark));
Expect.equals(
'aXXcaXXdae',
r('abcabdae').replaceAllMapped(RegExp('b'), (_) => a('XX')),
);
Expect.equals(
'aXXcaXXdae',
r('abcabdae').replaceAllMapped(RegExpWrap('b'), (_) => a('XX')),
);
}
void testCompareTo(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
Expect.equals(0, r('').compareTo(a('')));
Expect.equals(0, r('str').compareTo(a('str')));
Expect.equals(-1, r('str').compareTo(a('string')));
Expect.equals(1, r('string').compareTo(a('str')));
Expect.equals(1, r('string').compareTo(a('')));
Expect.equals(-1, r('').compareTo(a('string')));
}
void testCharCodes() {
void test(String str) {
final list = str.codeUnits;
Expect.equals(str.length, list.length);
for (int i = 0; i < str.length; i++) {
Expect.equals(str.codeUnitAt(i), list[i]);
}
}
test(jsStringImpl('abc'));
test(jsStringImpl(''));
test(jsStringImpl(' '));
}
void testRepeat() {
List<String> testStrings = [
'',
'\x00',
'a',
'ab',
'\x80',
'\xff',
'\u2028',
'abcdef\u2028',
'\u{10002}',
'abcdef\u{10002}',
].map(jsStringImpl).toList();
List<int> counts = [0, 1, 2, 10];
void testRepeat(String str, int repeat) {
String expect;
if (repeat <= 0) {
expect = '';
} else if (repeat == 1) {
expect = str;
} else {
StringBuffer buf = StringBuffer();
for (int i = 0; i < repeat; i++) {
buf.write(str);
}
expect = buf.toString();
}
String actual = str * repeat;
Expect.equals(expect, actual, '$str#${str.length} * $repeat');
}
for (String str in testStrings) {
for (int repeat in counts) {
testRepeat(str, repeat);
}
}
}
void testPadLeft(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
Expect.equals(' 1', r('1').padLeft(5, a(' ')));
Expect.equals(' \u{10002}', r('\u{10002}').padLeft(5, a(' ')));
Expect.equals('', r('').padLeft(0, a('a')));
Expect.equals('a', r('').padLeft(1, a('a')));
Expect.equals('', r('').padLeft(-2, a('a')));
Expect.equals('xyzxyzxyzxyzxyz', r('').padLeft(5, a('xyz')));
Expect.equals('xyzxyzxyzxyza', r('a').padLeft(5, a('xyz')));
Expect.equals(
'\u{10002}\u{10002}\u{10002}aa',
r('aa').padLeft(5, a('\u{10002}')),
);
}
void testPadRight(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
Expect.equals('1 ', r('1').padRight(5, a(' ')));
Expect.equals('\u{10002} ', r('\u{10002}').padRight(5, a(' ')));
Expect.equals('', r('').padRight(0, a('a')));
Expect.equals('a', r('').padRight(1, a('a')));
Expect.equals('', r('').padRight(-2, a('a')));
Expect.equals('xyzxyzxyzxyzxyz', r('').padRight(5, a('xyz')));
Expect.equals('axyzxyzxyzxyz', r('a').padRight(5, a('xyz')));
Expect.equals(
'aa\u{10002}\u{10002}\u{10002}',
r('aa').padRight(5, a('\u{10002}')),
);
Expect.equals('a', r('a').padRight(10, a('')));
}
// Characters with Whitespace property (Unicode 6.3).
// 0009..000D ; White_Space # Cc <control-0009>..<control-000D>
// 0020 ; White_Space # Zs SPACE
// 0085 ; White_Space # Cc <control-0085>
// 00A0 ; White_Space # Zs NO-BREAK SPACE
// 1680 ; White_Space # Zs OGHAM SPACE MARK
// 2000..200A ; White_Space # Zs EN QUAD..HAIR SPACE
// 2028 ; White_Space # Zl LINE SEPARATOR
// 2029 ; White_Space # Zp PARAGRAPH SEPARATOR
// 202F ; White_Space # Zs NARROW NO-BREAK SPACE
// 205F ; White_Space # Zs MEDIUM MATHEMATICAL SPACE
// 3000 ; White_Space # Zs IDEOGRAPHIC SPACE
// And BOM:
// FEFF ; Byte order mark.
const whitespace = const [
0x09,
0x0A,
0x0B,
0x0C,
0x0D,
0x20,
0x85,
0xA0,
0x1680,
0x2000,
0x2001,
0x2002,
0x2003,
0x2004,
0x2005,
0x2006,
0x2007,
0x2008,
0x2009,
0x200A,
0x2028,
0x2029,
0x202F,
0x205F,
0x3000,
0xFEFF,
];
void testTrimTest() {
String s(String a) => jsStringImpl(a);
Expect.equals('', s(' ').trim());
Expect.equals('', s(' ').trim());
Expect.equals('left', s(' left').trim());
Expect.equals('right', s('right ').trim());
Expect.equals('', ' \t \n \r '.trim());
for (final ws in whitespace) {
Expect.equals('', s(String.fromCharCode(ws)).trim());
}
}
void testTrimLeftTest() {
String s(String a) => jsStringImpl(a);
Expect.equals('', s(' ').trimLeft());
Expect.equals('', s(' ').trimLeft());
Expect.equals('left', s(' left').trimLeft());
Expect.equals('right ', s('right ').trimLeft());
Expect.equals('', ' \t \n \r '.trimLeft());
for (final ws in whitespace) {
Expect.equals('', s(String.fromCharCode(ws)).trimLeft());
}
}
void testTrimRightTest() {
String s(String a) => jsStringImpl(a);
Expect.equals('', s(' ').trimRight());
Expect.equals('', s(' ').trimRight());
Expect.equals(' left', s(' left').trimRight());
Expect.equals('right', s('right ').trimRight());
Expect.equals('', ' \t \n \r '.trimRight());
for (final ws in whitespace) {
Expect.equals('', s(String.fromCharCode(ws)).trimRight());
}
}
void testMatch(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
String astr = a('this is a string with hello here and hello there');
// No match:
{
final helloPattern = r('with (hello)');
final matches = helloPattern.allMatches(astr);
Expect.isFalse(matches.iterator.moveNext());
}
int check(String helloPattern, Iterable<Match> matches, String expected) {
int count = 0;
int start = 0;
for (final match in matches) {
Expect.equals(astr.indexOf(expected, start), match.start);
Expect.equals(
astr.indexOf(expected, start) + helloPattern.length,
match.end,
);
Expect.equals(helloPattern, match.pattern);
Expect.equals(astr, match.input);
Expect.equals(helloPattern, match[0]);
Expect.equals(0, match.groupCount);
count++;
start = match.end;
}
return count;
}
// One match:
{
String helloPattern = r('with hello');
Iterable<Match> matches = helloPattern.allMatches(astr);
final count = check(helloPattern, matches, a('with'));
Expect.equals(1, count);
}
// Two matches:
{
String helloPattern = r('hello');
Iterable<Match> matches = helloPattern.allMatches(astr);
final count = check(helloPattern, matches, a('hello'));
Expect.equals(2, count);
}
// Empty pattern:
{
String pattern = r('');
Iterable<Match> matches = pattern.allMatches(astr);
Expect.isTrue(matches.iterator.moveNext());
}
// Empty string:
{
String pattern = r('foo');
String str = a('');
Iterable<Match> matches = pattern.allMatches(str);
Expect.isFalse(matches.iterator.moveNext());
}
// Empty pattern and string:
{
String pattern = r('');
String str = a('');
Iterable<Match> matches = pattern.allMatches(str);
Expect.isTrue(matches.iterator.moveNext());
}
// Match as prefix:
{
String pattern = r('an');
String str = a('banana');
Expect.isNull(pattern.matchAsPrefix(str));
Expect.isNull(pattern.matchAsPrefix(str, 0));
var m = pattern.matchAsPrefix(str, 1)!;
Expect.equals('an', m[0]);
Expect.equals(1, m.start);
Expect.isNull(pattern.matchAsPrefix(str, 2));
m = pattern.matchAsPrefix(str, 3)!;
Expect.equals('an', m[0]);
Expect.equals(3, m.start);
Expect.isNull(pattern.matchAsPrefix(str, 4));
Expect.isNull(pattern.matchAsPrefix(str, 5));
Expect.isNull(pattern.matchAsPrefix(str, 6));
Expect.throws(() => pattern.matchAsPrefix(str, -1));
Expect.throws(() => pattern.matchAsPrefix(str, 7));
}
// Start:
{
String p = r('ass');
String s = a('assassin');
Expect.equals(2, p.allMatches(s).length);
Expect.equals(2, p.allMatches(s, 0).length);
Expect.equals(1, p.allMatches(s, 1).length);
Expect.equals(0, p.allMatches(s, 4).length);
Expect.equals(0, p.allMatches(s, s.length).length);
Expect.throws(() => p.allMatches(s, -1));
Expect.throws(() => p.allMatches(s, s.length + 1));
}
}
void testSubstring() {
String s(String a) => jsStringImpl(a);
final emptyStr = s('');
Expect.equals(emptyStr.substring(0), '');
Expect.throwsRangeError(() => emptyStr.substring(1));
Expect.throwsRangeError(() => emptyStr.substring(-1));
final abcStr = s('abc');
Expect.equals(abcStr.substring(0), 'abc');
Expect.equals(abcStr.substring(1), 'bc');
Expect.equals(abcStr.substring(2), 'c');
Expect.equals(abcStr.substring(3), '');
Expect.throwsRangeError(() => abcStr.substring(4));
Expect.throwsRangeError(() => abcStr.substring(-1));
// Test that providing null goes to the end.
Expect.equals(emptyStr.substring(0, null), '');
Expect.throwsRangeError(() => emptyStr.substring(1, null));
Expect.throwsRangeError(() => emptyStr.substring(-1, null));
Expect.equals(abcStr.substring(0, null), 'abc');
Expect.equals(abcStr.substring(1, null), 'bc');
Expect.equals(abcStr.substring(2, null), 'c');
Expect.equals(abcStr.substring(3, null), '');
Expect.throwsRangeError(() => abcStr.substring(4, null));
Expect.throwsRangeError(() => abcStr.substring(-1, null));
}
void testSplitString(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
Expect.listEquals(['a', 'b', 'c'], r('a b c').split(a(' ')));
Expect.listEquals(['a', 'b', 'c'], r('adbdc').split(a('d')));
Expect.listEquals(['abc'], r('abc').split(a(' ')));
Expect.listEquals(['a', 'b', 'c'], a('abc').split(a('')));
}
void testSplit(TestMode mode, Pattern p(String s)) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
Expect.listEquals(['a', 'b', 'c'], r('a b c').split(p(' ')));
Expect.listEquals(['a', 'b', 'c'], r('adbdc').split(p(r'[dz]')));
}
void testSplitRegExp(TestMode mode) => testSplit(mode, (s) => RegExp(s));
void testSplitUserPattern(TestMode mode) =>
testSplit(mode, (s) => RegExpWrap(s));
void testCase() {
String s(String a) => jsStringImpl(a);
Expect.equals('ABC', s('aBc').toUpperCase());
Expect.equals('abc', s('AbC').toLowerCase());
}
void testReplace(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
// Test replaceFirst
Expect.equals(
'AtoBtoCDtoE',
r('AfromBtoCDtoE').replaceFirst(a('from'), a('to')),
);
Expect.equals(
'toABtoCDtoE',
r('fromABtoCDtoE').replaceFirst(a('from'), a('to')),
);
Expect.equals(
'toABtoCDtoEto',
r('fromABtoCDtoEto').replaceFirst(a('from'), a('to')),
);
Expect.equals('ABC', r('ABC').replaceFirst(a('from'), a('to')));
Expect.equals('', r('').replaceFirst(a('from'), a('to')));
Expect.equals(
'foo-AAA-foo-bar',
r('foo-bar-foo-bar').replaceFirst(a('bar'), a('AAA'), 4),
);
Expect.equals(
'foo-bar-foo-bar',
r('foo-bar-foo-bar').replaceFirst(RegExp(r'^foo'), a(''), 8),
);
Expect.throwsRangeError(() => r('hello').replaceFirst(a('h'), a('X'), -1));
Expect.throwsRangeError(() => r('hello').replaceFirst(a('h'), a('X'), 6));
// Test replaceFirstMapped.
Expect.equals(
'AtoBtoCDtoE',
r('AfromBtoCDtoE').replaceFirstMapped(a('from'), (_) => a('to')),
);
Expect.equals('ABC', r('ABC').replaceFirstMapped(a('from'), (_) => a('to')));
Expect.equals('', r('').replaceFirstMapped(a('from'), (_) => a('to')));
Expect.equals(
'foo-AAA-foo-bar',
r('foo-bar-foo-bar').replaceFirstMapped(a('bar'), (_) => a('AAA'), 4),
);
Expect.equals(
'foo-bar-foo-bar',
r('foo-bar-foo-bar').replaceFirstMapped(RegExp(r'^foo'), (_) => a(''), 8),
);
Expect.throwsRangeError(
() => r('hello').replaceFirstMapped(a('h'), (_) => a('X'), -1),
);
Expect.throwsRangeError(
() => r('hello').replaceFirstMapped(a('h'), (_) => a('X'), 6),
);
Expect.equals(
'foo-BAR-foo-bar',
r(
'foo-bar-foo-bar',
).replaceFirstMapped(a('bar'), (v) => a(v[0]!.toUpperCase())),
);
for (final string in [r(''), r('x'), r('foo'), r('x\u2000z')]) {
for (final replacement in [a(''), a('foo'), a(string)]) {
for (int start = 0; start <= string.length; start++) {
var expect;
for (int end = start; end <= string.length; end++) {
expect =
string.substring(0, start) + replacement + string.substring(end);
Expect.equals(
expect,
string.replaceRange(start, end, replacement),
"'$string'[$start:$end]='$replacement'",
);
}
// Reuse expect from 'end == string.length' case when omitting end.
Expect.equals(
expect,
string.replaceRange(start, null, replacement),
"'$string'[$start:]='$replacement'",
);
}
}
Expect.throws(() => string.replaceRange(-1, 0, 'x'));
Expect.throws(() => string.replaceRange(0, string.length + 1, 'x'));
}
}
void testSplitMapJoin(TestMode mode) {
String r(String s) => getStr(s, Position.jsStringImplReceiver, mode);
String a(String s) => getStr(s, Position.jsStringImplArgument, mode);
String mark(Match m) => a('[${m[0]}]');
String rest(String s) => a('<${s}>');
Expect.equals(
'<a>[b]<ca>[b]<dae>',
r('abcabdae').splitMapJoin(a('b'), onMatch: mark, onNonMatch: rest),
);
Expect.equals(
'<abcabdae>',
r('abcabdae').splitMapJoin(a('f'), onMatch: mark, onNonMatch: rest),
);
Expect.equals(
'<>',
r('').splitMapJoin(a('from'), onMatch: mark, onNonMatch: rest),
);
Expect.equals(
'<>[]<>',
r('').splitMapJoin(a(''), onMatch: mark, onNonMatch: rest),
);
Expect.equals(
'<>[]<A>[]<B>[]<C>[]<>',
r('ABC').splitMapJoin(a(''), onMatch: mark, onNonMatch: rest),
);
Expect.equals(
'[a]bc[a]bd[a]e',
r('abcabdae').splitMapJoin(a('a'), onMatch: mark),
);
Expect.equals(
'<>a<bc>a<bd>a<e>',
r('abcabdae').splitMapJoin(a('a'), onNonMatch: rest),
);
}
void main() {
for (final mode in [
TestMode.jsStringImplReceiver,
TestMode.jsStringImplArgument,
TestMode.jsStringImplReceiverAndArguments,
]) {
testLength(mode);
testConcat(mode);
testEquals(mode);
testEndsWith(mode);
testStartsWith(mode);
testIndexOf(mode);
testLastIndexOf(mode);
testContains(mode);
testReplaceAll(mode);
testReplaceAllMapped(mode);
testCompareTo(mode);
testPadLeft(mode);
testPadRight(mode);
testMatch(mode);
testSplitString(mode);
testSplitRegExp(mode);
testSplitUserPattern(mode);
testReplace(mode);
testSplitMapJoin(mode);
}
testOutOfRange();
testIndex();
testCodeUnitAt();
testHashCode();
testCharCodes();
testRepeat();
testTrimTest();
testTrimLeftTest();
testTrimRightTest();
testSubstring();
testCase();
}
// A Pattern implementation with the same capabilities as a RegExp, but not
// directly recognizable as a RegExp.
class RegExpWrap implements Pattern {
final RegExp regexp;
RegExpWrap(String source) : regexp = RegExp(source);
Iterable<Match> allMatches(String string, [int start = 0]) =>
regexp.allMatches(string, start);
Match? matchAsPrefix(String string, [int start = 0]) =>
regexp.matchAsPrefix(string, start);
String toString() => 'Wrap(/${regexp.pattern}/)';
}