2008-03-05 09:52:00 -05:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
// The MatchCollection lists the successful matches that
|
|
|
|
// result when searching a string for a regular expression.
|
|
|
|
//
|
|
|
|
|
2008-11-17 18:29:00 -05:00
|
|
|
namespace System.Text.RegularExpressions
|
|
|
|
{
|
2008-03-05 09:52:00 -05:00
|
|
|
|
|
|
|
using System.Collections;
|
|
|
|
|
|
|
|
|
|
|
|
// This collection returns a sequence of successful match results, either
|
|
|
|
// from GetMatchCollection() or GetExecuteCollection(). It stops when the
|
|
|
|
// first failure is encountered (it does not return the failed match).
|
|
|
|
//| <include file='doc\RegexMatchCollection.uex' path='docs/doc[@for="MatchCollection"]/*' />
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>
|
|
|
|
/// Represents the set of names appearing as capturing group
|
|
|
|
/// names in a regular expression.
|
|
|
|
/// </para>
|
|
|
|
/// </devdoc>
|
2008-11-17 18:29:00 -05:00
|
|
|
//[ Serializable() ]
|
2008-03-05 09:52:00 -05:00
|
|
|
public class MatchCollection : ICollection {
|
|
|
|
internal Regex _regex;
|
|
|
|
internal ArrayList _matches;
|
|
|
|
internal bool _done;
|
|
|
|
internal String _input;
|
|
|
|
internal int _beginning;
|
|
|
|
internal int _length;
|
|
|
|
internal int _startat;
|
|
|
|
internal int _prevlen;
|
|
|
|
|
|
|
|
private static int infinite = 0x7FFFFFFF;
|
|
|
|
|
|
|
|
// Nonpublic constructor
|
|
|
|
internal MatchCollection(Regex regex, String input, int beginning, int length, int startat) {
|
|
|
|
if (startat < 0 || startat > input.Length)
|
|
|
|
throw new ArgumentOutOfRangeException("startat", "Begin index not negative");//XXX: SR.GetString(SR.BeginIndexNotNegative));
|
|
|
|
|
|
|
|
_regex = regex;
|
|
|
|
_input = input;
|
|
|
|
_beginning = beginning;
|
|
|
|
_length = length;
|
|
|
|
_startat = startat;
|
|
|
|
_prevlen = -1;
|
|
|
|
_matches = new ArrayList();
|
|
|
|
_done = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal Match GetMatch(int i) {
|
|
|
|
if (i < 0)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
if (_matches.Count > i)
|
|
|
|
return(Match)_matches[i];
|
|
|
|
|
|
|
|
if (_done)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
Match match;
|
|
|
|
|
|
|
|
do {
|
|
|
|
match = _regex.Run(false, _prevlen, _input, _beginning, _length, _startat);
|
|
|
|
|
|
|
|
if (!match.Success) {
|
|
|
|
_done = true;
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
_matches.Add(match);
|
|
|
|
|
|
|
|
_prevlen = match._length;
|
|
|
|
_startat = match._textpos;
|
|
|
|
|
|
|
|
} while (_matches.Count <= i);
|
|
|
|
|
|
|
|
return match;
|
|
|
|
}
|
|
|
|
|
|
|
|
//| <include file='doc\RegexMatchCollection.uex' path='docs/doc[@for="MatchCollection.Count"]/*' />
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>
|
|
|
|
/// Returns the number of captures.
|
|
|
|
/// </para>
|
|
|
|
/// </devdoc>
|
|
|
|
public int Count {
|
|
|
|
get {
|
|
|
|
if (_done)
|
|
|
|
return _matches.Count;
|
|
|
|
|
|
|
|
GetMatch(infinite);
|
|
|
|
|
|
|
|
return _matches.Count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//| <include file='doc\RegexMatchCollection.uex' path='docs/doc[@for="MatchCollection.SyncRoot"]/*' />
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public Object SyncRoot {
|
|
|
|
get {
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//| <include file='doc\RegexMatchCollection.uex' path='docs/doc[@for="MatchCollection.IsSynchronized"]/*' />
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public bool IsSynchronized {
|
|
|
|
get {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//| <include file='doc\RegexMatchCollection.uex' path='docs/doc[@for="MatchCollection.IsReadOnly"]/*' />
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public bool IsReadOnly {
|
|
|
|
get {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//| <include file='doc\RegexMatchCollection.uex' path='docs/doc[@for="MatchCollection.this"]/*' />
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>
|
|
|
|
/// Returns the ith Match in the collection.
|
|
|
|
/// </para>
|
|
|
|
/// </devdoc>
|
|
|
|
public virtual Match this[int i]
|
|
|
|
{
|
|
|
|
get {
|
|
|
|
Match match;
|
|
|
|
|
|
|
|
match = GetMatch(i);
|
|
|
|
|
|
|
|
if (match == null)
|
|
|
|
throw new ArgumentOutOfRangeException("i");
|
|
|
|
|
|
|
|
return match;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//| <include file='doc\RegexMatchCollection.uex' path='docs/doc[@for="MatchCollection.CopyTo"]/*' />
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>
|
|
|
|
/// Copies all the elements of the collection to the given array
|
|
|
|
/// starting at the given index.
|
|
|
|
/// </para>
|
|
|
|
/// </devdoc>
|
|
|
|
public void CopyTo(Array array, int arrayIndex) {
|
|
|
|
// property access to force computation of whole array
|
|
|
|
int count = Count;
|
|
|
|
|
|
|
|
_matches.CopyTo(array, arrayIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
//| <include file='doc\RegexMatchCollection.uex' path='docs/doc[@for="MatchCollection.GetEnumerator"]/*' />
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>
|
|
|
|
/// Provides an enumerator in the same order as Item[i].
|
|
|
|
/// </para>
|
|
|
|
/// </devdoc>
|
|
|
|
public IEnumerator GetEnumerator() {
|
|
|
|
return new MatchEnumerator(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// This non-public enumerator lists all the group matches.
|
|
|
|
// Should it be public?
|
2008-11-17 18:29:00 -05:00
|
|
|
//[ Serializable() ]
|
2008-03-05 09:52:00 -05:00
|
|
|
internal class MatchEnumerator : IEnumerator {
|
|
|
|
internal MatchCollection _matchcoll;
|
|
|
|
internal Match _match = null;
|
|
|
|
internal int _curindex;
|
|
|
|
internal bool _done;
|
|
|
|
|
|
|
|
// Nonpublic constructor
|
|
|
|
internal MatchEnumerator(MatchCollection matchcoll) {
|
|
|
|
_matchcoll = matchcoll;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Advance to the next match
|
|
|
|
public bool MoveNext() {
|
|
|
|
if (_done)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
_match = _matchcoll.GetMatch(_curindex++);
|
|
|
|
|
|
|
|
if (_match == null) {
|
|
|
|
_done = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The current match
|
|
|
|
public Object Current {
|
2008-11-17 18:29:00 -05:00
|
|
|
get {
|
2008-03-05 09:52:00 -05:00
|
|
|
if (_match == null)
|
|
|
|
throw new InvalidOperationException("Enum not Started");//XXX: SR.GetString(SR.EnumNotStarted));
|
|
|
|
return _match;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// The current match
|
|
|
|
public Match Match {
|
|
|
|
get {
|
|
|
|
return _match;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Position before the first item
|
|
|
|
public void Reset() {
|
|
|
|
_curindex = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|