os/textandloc/textandlocutils/numbergrouping/src/StateMachine.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
* Copyright (c) 2002 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     3
* All rights reserved.
sl@0
     4
* This component and the accompanying materials are made available
sl@0
     5
* under the terms of the License "Eclipse Public License v1.0"
sl@0
     6
* which accompanies this distribution, and is available
sl@0
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     8
*
sl@0
     9
* Initial Contributors:
sl@0
    10
* Nokia Corporation - initial contribution.
sl@0
    11
*
sl@0
    12
* Contributors:
sl@0
    13
*
sl@0
    14
* Description: 
sl@0
    15
*
sl@0
    16
*/
sl@0
    17
sl@0
    18
sl@0
    19
#include "StateMachine.h"
sl@0
    20
sl@0
    21
// CStateMachine
sl@0
    22
CStateMachine::CStateMachine(TInt aMaxNumberChars, TInt aMaxNumberStates) : iMaxNumberChars(static_cast<TInt>(aMaxNumberChars)),
sl@0
    23
																			iMaxNumberStates(static_cast<TInt>(aMaxNumberStates))
sl@0
    24
{
sl@0
    25
}
sl@0
    26
sl@0
    27
CStateMachine* CStateMachine::NewL(TInt aMaxNumberChars, TInt aMaxNumberStates)
sl@0
    28
{
sl@0
    29
	CStateMachine* s = NewLC(aMaxNumberChars, aMaxNumberStates);
sl@0
    30
	CleanupStack::Pop();
sl@0
    31
	return s;
sl@0
    32
}
sl@0
    33
sl@0
    34
CStateMachine* CStateMachine::NewLC(TInt aMaxNumberChars, TInt aMaxNumberStates)
sl@0
    35
{
sl@0
    36
	CStateMachine* s = new(ELeave)CStateMachine(aMaxNumberChars, aMaxNumberStates);
sl@0
    37
	CleanupStack::PushL(s);
sl@0
    38
	s->ConstructL();
sl@0
    39
	return s;
sl@0
    40
}
sl@0
    41
sl@0
    42
CStateMachine::~CStateMachine()
sl@0
    43
{
sl@0
    44
	for(TInt i = 0; i < iMaxNumberChars; ++i)
sl@0
    45
		delete iStateTable[i];
sl@0
    46
	
sl@0
    47
	delete iStateTable;
sl@0
    48
}
sl@0
    49
sl@0
    50
void CStateMachine::ConstructL()
sl@0
    51
{
sl@0
    52
	iStateTable = new TInt*[iMaxNumberChars];	
sl@0
    53
sl@0
    54
	for(TInt i = 0; i < iMaxNumberChars; ++i)
sl@0
    55
	{
sl@0
    56
		iStateTable[i] = new TInt[iMaxNumberStates];
sl@0
    57
		for(TInt j = 0; j < iMaxNumberStates; j++)
sl@0
    58
			iStateTable[i][j] = 0;
sl@0
    59
	}
sl@0
    60
}
sl@0
    61
sl@0
    62
void CStateMachine::AddStateTransistionL(TChar aChar, TInt aState, TInt aNextState)
sl@0
    63
{
sl@0
    64
	RArray<TInt> Dummy;
sl@0
    65
	TInt CharIndex = MapIndex(aChar, Dummy);
sl@0
    66
	Dummy.Close();
sl@0
    67
	
sl@0
    68
	AddStateTransistionL(CharIndex, aState, aNextState);
sl@0
    69
}
sl@0
    70
sl@0
    71
void CStateMachine::AddStateTransistionL(TInt aIndex, TInt aState, TInt aNextState)
sl@0
    72
{
sl@0
    73
    if(aIndex < 0 || aIndex > iMaxNumberChars || aState > iMaxNumberStates || aNextState > iMaxNumberStates)
sl@0
    74
		User::Leave(KErrGeneral);
sl@0
    75
sl@0
    76
	iStateTable[static_cast<TInt>(aIndex)][static_cast<TInt>(aState)] = static_cast<TInt>(aNextState);
sl@0
    77
}
sl@0
    78
sl@0
    79
TBool CStateMachine::Matches(const TDesC& aString)
sl@0
    80
{
sl@0
    81
	// Walk the state table and return true if the string matches the pattern.
sl@0
    82
	
sl@0
    83
	TInt			nTableState = KStateNoMatch;
sl@0
    84
	RArray<TInt>	arrIndices;
sl@0
    85
	TBool			bFound = EFalse;
sl@0
    86
sl@0
    87
	for(TInt nStringIndex = 0; nStringIndex < aString.Length(); ++nStringIndex)
sl@0
    88
	{
sl@0
    89
		arrIndices.Reset();
sl@0
    90
		
sl@0
    91
		TChar	charChar = aString[nStringIndex];
sl@0
    92
		TInt	nTableIndex = MapIndex(charChar, arrIndices);
sl@0
    93
sl@0
    94
		if(nTableIndex == -1) 
sl@0
    95
			return EFalse;
sl@0
    96
		
sl@0
    97
		TInt nCount = arrIndices.Count();
sl@0
    98
		
sl@0
    99
		if(nCount)
sl@0
   100
		{
sl@0
   101
			for(TInt i = 0; i < nCount; i++)
sl@0
   102
			{
sl@0
   103
				nTableIndex = arrIndices[i];
sl@0
   104
				TInt NewTableState = iStateTable[nTableIndex][nTableState];
sl@0
   105
				
sl@0
   106
				if( NewTableState != KStateNoMatch || 
sl@0
   107
					(NewTableState == KStateNoMatch && i == (arrIndices.Count() - 1)) ||
sl@0
   108
					NewTableState == KStateMatched)
sl@0
   109
				{
sl@0
   110
					nTableState = NewTableState;
sl@0
   111
					break;
sl@0
   112
				}
sl@0
   113
			}
sl@0
   114
		}
sl@0
   115
		else
sl@0
   116
		{
sl@0
   117
			nTableState = iStateTable[nTableIndex][nTableState];
sl@0
   118
		}
sl@0
   119
sl@0
   120
		if(nTableState == KStateNoMatch)
sl@0
   121
			break;
sl@0
   122
sl@0
   123
		if(nTableState == KStateMatched)
sl@0
   124
		{
sl@0
   125
			bFound = ETrue;
sl@0
   126
			break;
sl@0
   127
		}
sl@0
   128
	}
sl@0
   129
sl@0
   130
	arrIndices.Close();
sl@0
   131
	
sl@0
   132
	return bFound;
sl@0
   133
}
sl@0
   134
sl@0
   135
TInt CStateMachine::MapIndex(TChar aChar, RArray<TInt>& aIndices)
sl@0
   136
{
sl@0
   137
    // Check the type of aChar and return the relevant index or indicies
sl@0
   138
	if(aChar.IsDigit())
sl@0
   139
	{
sl@0
   140
		TInt nIndex = static_cast<TInt>(aChar.GetNumericValue());
sl@0
   141
		
sl@0
   142
		TInt ret = KErrNone;
sl@0
   143
		ret |= aIndices.Append(nIndex);
sl@0
   144
		ret |= aIndices.Append(KCharacterDot);
sl@0
   145
		__ASSERT_DEBUG(!ret, User::Panic(_L("RArray append failure"), ret));
sl@0
   146
		if (KErrNone != ret) return ret;
sl@0
   147
sl@0
   148
		return nIndex;
sl@0
   149
	}
sl@0
   150
sl@0
   151
	if(aChar == '+')	return KCharacterPlus;
sl@0
   152
	if(aChar == '.')	return KCharacterDot;
sl@0
   153
sl@0
   154
    return -1;  // TO DO : define 
sl@0
   155
}
sl@0
   156
sl@0
   157
void CStateMachine::GetWildcardVersionOfPattern( 
sl@0
   158
    TText aWildcardChar, 
sl@0
   159
    TDes& aWildcardedPattern ) const
sl@0
   160
    {
sl@0
   161
    aWildcardedPattern.SetLength(0);
sl@0
   162
    TInt maxLength = aWildcardedPattern.MaxLength();
sl@0
   163
    // There is a column in the StateTable for each character in the regular expression. The first character
sl@0
   164
    // of the regexp in column [0], last in column [Length()-1]
sl@0
   165
    // The non-zero values found within a column of the StateTable represent the characters
sl@0
   166
    // that are valid at that position in a candidate match.
sl@0
   167
    // An example pattern is calculated by examining the StateTable.
sl@0
   168
    // For each column, count the number of matching ( i.e. not KStateNoMatch) states
sl@0
   169
    // If only one, then place that character in the example pattern.
sl@0
   170
    // If more than one, put the wildcard in.
sl@0
   171
    for(TInt stateIndex = 0; stateIndex < iMaxNumberStates && stateIndex < maxLength; stateIndex++)
sl@0
   172
        {
sl@0
   173
        TInt matches = 0;
sl@0
   174
        TInt matchedIndex = -1;
sl@0
   175
        for (TInt charIndex = 0; charIndex < iMaxNumberChars; charIndex++ )
sl@0
   176
            {
sl@0
   177
            TInt nextState = iStateTable[charIndex][stateIndex];
sl@0
   178
sl@0
   179
            if ( nextState != KStateNoMatch )
sl@0
   180
                {
sl@0
   181
                matches++;
sl@0
   182
                matchedIndex = charIndex;
sl@0
   183
                if (matches > 1 )
sl@0
   184
                    break;
sl@0
   185
                }
sl@0
   186
            }
sl@0
   187
        if ( matches == 0 ) // Have found an empty column.  Unused part of state machine. Stop filling
sl@0
   188
            {
sl@0
   189
            break;
sl@0
   190
            }
sl@0
   191
        else if ( matches > 1 )
sl@0
   192
            aWildcardedPattern.Append(aWildcardChar);
sl@0
   193
        else
sl@0
   194
            {
sl@0
   195
            if ( 0 <= matchedIndex && matchedIndex <= 9 ) // Must be a numeric digit
sl@0
   196
                {
sl@0
   197
                aWildcardedPattern.Append((TText)(matchedIndex+'0'));
sl@0
   198
                }
sl@0
   199
            else if ( matchedIndex == KCharacterDot )
sl@0
   200
                {
sl@0
   201
                aWildcardedPattern.Append(aWildcardChar);
sl@0
   202
                }
sl@0
   203
            else if ( matchedIndex == KCharacterPlus )
sl@0
   204
                {
sl@0
   205
                aWildcardedPattern.Append('+');
sl@0
   206
                }
sl@0
   207
            else
sl@0
   208
                {
sl@0
   209
                aWildcardedPattern.Append(aWildcardChar);
sl@0
   210
                }
sl@0
   211
            }
sl@0
   212
        }
sl@0
   213
sl@0
   214
    }
sl@0
   215
// End of File
sl@0
   216