blob: 7c9438131082e89df4f2d056c7ca79e62c10e3f6 [file] [log] [blame]
// Copyright (c) 2012, 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.
patch class RegExp {
/* patch */ factory RegExp(String source,
{bool multiLine: false,
bool caseSensitive: true}) {
return new _JSSyntaxRegExp(source,
multiLine: multiLine,
caseSensitive: caseSensitive);
}
}
class _JSRegExpMatch implements Match {
_JSRegExpMatch(this._regexp, this.input, this._match);
int get start => _start(0);
int get end => _end(0);
int _start(int groupIdx) {
return _match[(groupIdx * _MATCH_PAIR)];
}
int _end(int groupIdx) {
return _match[(groupIdx * _MATCH_PAIR) + 1];
}
String group(int groupIdx) {
if (groupIdx < 0 || groupIdx > _regexp._groupCount) {
throw new RangeError.value(groupIdx);
}
int startIndex = _start(groupIdx);
int endIndex = _end(groupIdx);
if (startIndex == -1) {
assert(endIndex == -1);
return null;
}
return input._substringUnchecked(startIndex, endIndex);
}
String operator [](int groupIdx) {
return this.group(groupIdx);
}
List<String> groups(List<int> groupsSpec) {
var groupsList = new List<String>(groupsSpec.length);
for (int i = 0; i < groupsSpec.length; i++) {
groupsList[i] = group(groupsSpec[i]);
}
return groupsList;
}
int get groupCount => _regexp._groupCount;
Pattern get pattern => _regexp;
final RegExp _regexp;
final String input;
final List<int> _match;
static const int _MATCH_PAIR = 2;
}
class _JSSyntaxRegExp implements RegExp {
factory _JSSyntaxRegExp(
String pattern,
{bool multiLine: false,
bool caseSensitive: true}) native "JSSyntaxRegExp_factory";
Match firstMatch(String str) {
List match = _ExecuteMatch(str, 0);
if (match == null) {
return null;
}
return new _JSRegExpMatch(this, str, match);
}
Iterable<Match> allMatches(String string, [int start = 0]) {
if (string is! String) throw new ArgumentError(string);
if (start is! int) throw new ArgumentError(start);
if (0 > start || start > string.length) {
throw new RangeError.range(start, 0, string.length);
}
return new _AllMatchesIterable(this, string, start);
}
Match matchAsPrefix(String string, [int start = 0]) {
if (start < 0 || start > string.length) {
throw new RangeError.range(start, 0, string.length);
}
// Inefficient check that searches for a later match too.
// Change this when possible.
List<int> list = _ExecuteMatch(string, start);
if (list == null) return null;
if (list[0] != start) return null;
return new _JSRegExpMatch(this, string, list);
}
bool hasMatch(String str) {
List match = _ExecuteMatch(str, 0);
return (match == null) ? false : true;
}
String stringMatch(String str) {
List match = _ExecuteMatch(str, 0);
if (match == null) {
return null;
}
return str._substringUnchecked(match[0], match[1]);
}
String get pattern native "JSSyntaxRegExp_getPattern";
bool get isMultiLine native "JSSyntaxRegExp_getIsMultiLine";
bool get isCaseSensitive native "JSSyntaxRegExp_getIsCaseSensitive";
int get _groupCount native "JSSyntaxRegExp_getGroupCount";
List _ExecuteMatch(String str, int start_index)
native "JSSyntaxRegExp_ExecuteMatch";
}
class _AllMatchesIterable extends IterableBase<Match> {
final _JSSyntaxRegExp _re;
final String _str;
final int _start;
_AllMatchesIterable(this._re, this._str, this._start);
Iterator<Match> get iterator => new _AllMatchesIterator(_re, _str, _start);
}
class _AllMatchesIterator implements Iterator<Match> {
final String _str;
int _nextIndex;
_JSSyntaxRegExp _re;
Match _current;
_AllMatchesIterator(this._re, this._str, this._nextIndex);
Match get current => _current;
bool moveNext() {
if (_re == null) return false; // Cleared after a failed match.
if (_nextIndex <= _str.length) {
var match = _re._ExecuteMatch(_str, _nextIndex);
if (match != null) {
_current = new _JSRegExpMatch(_re, _str, match);
_nextIndex = _current.end;
if (_nextIndex == _current.start) {
// Zero-width match. Advance by one more.
_nextIndex++;
}
return true;
}
}
_current = null;
_re = null;
return false;
}
}