Update contrib.
2 * Copyright (c) 2002 Nokia Corporation and/or its subsidiary(-ies).
4 * This component and the accompanying materials are made available
5 * under the terms of the License "Eclipse Public License v1.0"
6 * which accompanies this distribution, and is available
7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
19 #include "StateMachine.h"
22 CStateMachine::CStateMachine(TInt aMaxNumberChars, TInt aMaxNumberStates) : iMaxNumberChars(static_cast<TInt>(aMaxNumberChars)),
23 iMaxNumberStates(static_cast<TInt>(aMaxNumberStates))
27 CStateMachine* CStateMachine::NewL(TInt aMaxNumberChars, TInt aMaxNumberStates)
29 CStateMachine* s = NewLC(aMaxNumberChars, aMaxNumberStates);
34 CStateMachine* CStateMachine::NewLC(TInt aMaxNumberChars, TInt aMaxNumberStates)
36 CStateMachine* s = new(ELeave)CStateMachine(aMaxNumberChars, aMaxNumberStates);
37 CleanupStack::PushL(s);
42 CStateMachine::~CStateMachine()
44 for(TInt i = 0; i < iMaxNumberChars; ++i)
45 delete iStateTable[i];
50 void CStateMachine::ConstructL()
52 iStateTable = new TInt*[iMaxNumberChars];
54 for(TInt i = 0; i < iMaxNumberChars; ++i)
56 iStateTable[i] = new TInt[iMaxNumberStates];
57 for(TInt j = 0; j < iMaxNumberStates; j++)
58 iStateTable[i][j] = 0;
62 void CStateMachine::AddStateTransistionL(TChar aChar, TInt aState, TInt aNextState)
65 TInt CharIndex = MapIndex(aChar, Dummy);
68 AddStateTransistionL(CharIndex, aState, aNextState);
71 void CStateMachine::AddStateTransistionL(TInt aIndex, TInt aState, TInt aNextState)
73 if(aIndex < 0 || aIndex > iMaxNumberChars || aState > iMaxNumberStates || aNextState > iMaxNumberStates)
74 User::Leave(KErrGeneral);
76 iStateTable[static_cast<TInt>(aIndex)][static_cast<TInt>(aState)] = static_cast<TInt>(aNextState);
79 TBool CStateMachine::Matches(const TDesC& aString)
81 // Walk the state table and return true if the string matches the pattern.
83 TInt nTableState = KStateNoMatch;
84 RArray<TInt> arrIndices;
85 TBool bFound = EFalse;
87 for(TInt nStringIndex = 0; nStringIndex < aString.Length(); ++nStringIndex)
91 TChar charChar = aString[nStringIndex];
92 TInt nTableIndex = MapIndex(charChar, arrIndices);
97 TInt nCount = arrIndices.Count();
101 for(TInt i = 0; i < nCount; i++)
103 nTableIndex = arrIndices[i];
104 TInt NewTableState = iStateTable[nTableIndex][nTableState];
106 if( NewTableState != KStateNoMatch ||
107 (NewTableState == KStateNoMatch && i == (arrIndices.Count() - 1)) ||
108 NewTableState == KStateMatched)
110 nTableState = NewTableState;
117 nTableState = iStateTable[nTableIndex][nTableState];
120 if(nTableState == KStateNoMatch)
123 if(nTableState == KStateMatched)
135 TInt CStateMachine::MapIndex(TChar aChar, RArray<TInt>& aIndices)
137 // Check the type of aChar and return the relevant index or indicies
140 TInt nIndex = static_cast<TInt>(aChar.GetNumericValue());
143 ret |= aIndices.Append(nIndex);
144 ret |= aIndices.Append(KCharacterDot);
145 __ASSERT_DEBUG(!ret, User::Panic(_L("RArray append failure"), ret));
146 if (KErrNone != ret) return ret;
151 if(aChar == '+') return KCharacterPlus;
152 if(aChar == '.') return KCharacterDot;
154 return -1; // TO DO : define
157 void CStateMachine::GetWildcardVersionOfPattern(
159 TDes& aWildcardedPattern ) const
161 aWildcardedPattern.SetLength(0);
162 TInt maxLength = aWildcardedPattern.MaxLength();
163 // There is a column in the StateTable for each character in the regular expression. The first character
164 // of the regexp in column [0], last in column [Length()-1]
165 // The non-zero values found within a column of the StateTable represent the characters
166 // that are valid at that position in a candidate match.
167 // An example pattern is calculated by examining the StateTable.
168 // For each column, count the number of matching ( i.e. not KStateNoMatch) states
169 // If only one, then place that character in the example pattern.
170 // If more than one, put the wildcard in.
171 for(TInt stateIndex = 0; stateIndex < iMaxNumberStates && stateIndex < maxLength; stateIndex++)
174 TInt matchedIndex = -1;
175 for (TInt charIndex = 0; charIndex < iMaxNumberChars; charIndex++ )
177 TInt nextState = iStateTable[charIndex][stateIndex];
179 if ( nextState != KStateNoMatch )
182 matchedIndex = charIndex;
187 if ( matches == 0 ) // Have found an empty column. Unused part of state machine. Stop filling
191 else if ( matches > 1 )
192 aWildcardedPattern.Append(aWildcardChar);
195 if ( 0 <= matchedIndex && matchedIndex <= 9 ) // Must be a numeric digit
197 aWildcardedPattern.Append((TText)(matchedIndex+'0'));
199 else if ( matchedIndex == KCharacterDot )
201 aWildcardedPattern.Append(aWildcardChar);
203 else if ( matchedIndex == KCharacterPlus )
205 aWildcardedPattern.Append('+');
209 aWildcardedPattern.Append(aWildcardChar);