os/textandloc/textrendering/texthandling/stext/TXTSCAN.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) 2003-2009 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 "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 "TXTETEXT.H"
sl@0
    20
#include "TXTSTD.H"
sl@0
    21
#include "OstTraceDefinitions.h"
sl@0
    22
#ifdef OST_TRACE_COMPILER_IN_USE
sl@0
    23
#include "TXTSCANTraces.h"
sl@0
    24
#endif
sl@0
    25
sl@0
    26
sl@0
    27
sl@0
    28
EXPORT_C TInt CPlainText::CharPosOfParagraph(TInt& aLength,TInt aParaOffset)const
sl@0
    29
/** Finds the length and the start position of a paragraph identified by its 
sl@0
    30
paragraph number the first paragraph is numbered zero.
sl@0
    31
sl@0
    32
Notes:
sl@0
    33
sl@0
    34
if aParaOffset is invalid, (equal to or greater than the total number of 
sl@0
    35
paragraphs), the function's return value is EScanEndOfData
sl@0
    36
sl@0
    37
@param aLength On return contains the length of the specified paragraph. 
sl@0
    38
@param aParaOffset The paragraph number. The first paragraph is numbered zero. 
sl@0
    39
@return The document position of the first character in the paragraph. */
sl@0
    40
	{
sl@0
    41
	__TEST_INVARIANT;
sl@0
    42
sl@0
    43
	TInt startPos=aLength=0;
sl@0
    44
	TUint scanMask=(EScanToUnitStart);
sl@0
    45
	aLength=ScanParas(startPos,scanMask);
sl@0
    46
	for (TInt offset=1;offset<=aParaOffset;offset++)
sl@0
    47
		{
sl@0
    48
		if (startPos<=EScanEndOfData)
sl@0
    49
			return EScanEndOfData;
sl@0
    50
		aLength=ScanParas(startPos,scanMask);
sl@0
    51
		}
sl@0
    52
	if (startPos==EScanEndOfData)
sl@0
    53
		startPos=iByteStore->Size()/sizeof(TText);
sl@0
    54
	return startPos-aLength;
sl@0
    55
	}
sl@0
    56
sl@0
    57
EXPORT_C TInt CPlainText::ParagraphNumberForPos(TInt& aPos)const
sl@0
    58
/** Gets the number of the paragraph which contains a document position. 
sl@0
    59
Paragraph numbering begins at zero.
sl@0
    60
sl@0
    61
@param aPos A document position. Must be valid or a panic occurs. On return, 
sl@0
    62
contains the document position of the first character in the paragraph in 
sl@0
    63
which it is located. 
sl@0
    64
@return The number of the paragraph containing the specified document position.
sl@0
    65
The first paragraph is numbered zero. */
sl@0
    66
	{
sl@0
    67
	__TEST_INVARIANT;
sl@0
    68
sl@0
    69
	if (aPos<0 || aPos>DocumentLength())
sl@0
    70
	    {
sl@0
    71
	    OstTrace0( TRACE_FATAL, CPLAINTEXT_PARAGRAPHNUMBERFORPOS, "ECharPosBeyondDocument" );
sl@0
    72
	    }
sl@0
    73
	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
sl@0
    74
	TUint scanMask=(EScanBackwards|EScanStayIfBoundary|EScanToUnitStart);
sl@0
    75
	ScanParas(aPos,scanMask);
sl@0
    76
	// aPos now holds the 1st char of the paragraph.
sl@0
    77
	TInt paraOffset=0,currentPos=0;
sl@0
    78
	scanMask=EScanToUnitStart;
sl@0
    79
	while (currentPos<aPos)
sl@0
    80
		{// Move forwards, counting paragraphs until we get to the current position.
sl@0
    81
		paraOffset++;
sl@0
    82
		ScanParas(currentPos,scanMask);
sl@0
    83
		}
sl@0
    84
	return paraOffset;
sl@0
    85
	}
sl@0
    86
sl@0
    87
EXPORT_C void CPlainText::GetWordInfo(TInt aCurrentPos,TInt& aStartPos,TInt& aLength,
sl@0
    88
									  TBool aPictureIsDelimiter,TBool aPunctuationIsDelimiter)const
sl@0
    89
/** Gets the start position and length of the word that contains the document 
sl@0
    90
position specified.
sl@0
    91
sl@0
    92
@param aCurrentPos A document position. Must be valid or a panic occurs. 
sl@0
    93
@param aStartPos On return, the document position of the first character in 
sl@0
    94
the word containing the position aCurrentPos.
sl@0
    95
@param aLength On return, the length of the word containing document position 
sl@0
    96
aCurrentPos.
sl@0
    97
@param aPictureIsDelimiter ETrue if picture characters should be treated as 
sl@0
    98
word delimiters, false if not. For example, when navigating text, this might 
sl@0
    99
be EFalse, but when spell checking, it might be ETrue. 
sl@0
   100
@param aPunctuationIsDelimiter ETrue if punctuation characters should be treated 
sl@0
   101
as word delimiters, EFalse if not. */
sl@0
   102
	{
sl@0
   103
	__TEST_INVARIANT;
sl@0
   104
sl@0
   105
	if (aCurrentPos<0 || aCurrentPos>DocumentLength())
sl@0
   106
	    {
sl@0
   107
	    OstTrace0( TRACE_FATAL, CPLAINTEXT_GETWORDINFO, "ECharPosBeyondDocument" );
sl@0
   108
	    }
sl@0
   109
	__ASSERT_ALWAYS(aCurrentPos>=0 && aCurrentPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
sl@0
   110
	aStartPos=aLength=aCurrentPos;
sl@0
   111
	// First find the start of the word.
sl@0
   112
	TUint scanMask=(EScanBackwards|
sl@0
   113
					EScanStayIfBoundary|
sl@0
   114
					EScanToUnitStart);
sl@0
   115
	if (aPictureIsDelimiter)
sl@0
   116
		scanMask|=EScanPictureIsDelimiter;
sl@0
   117
	if (aPunctuationIsDelimiter)
sl@0
   118
		scanMask|=EScanPunctuationIsDelimiter;
sl@0
   119
	ScanWords(aStartPos,scanMask);
sl@0
   120
	// Next find the end of the word.
sl@0
   121
	scanMask=(EScanStayIfBoundary|EScanToUnitEnd);
sl@0
   122
	if (aPictureIsDelimiter)
sl@0
   123
		scanMask|=EScanPictureIsDelimiter;
sl@0
   124
	if (aPunctuationIsDelimiter)
sl@0
   125
		scanMask|=EScanPunctuationIsDelimiter;
sl@0
   126
	ScanWords(aLength,scanMask);
sl@0
   127
	aLength-=aStartPos;
sl@0
   128
	if (aLength+aStartPos>(DocumentLength()+1))
sl@0
   129
	    {
sl@0
   130
	    OstTrace0( TRACE_DUMP, DUP2_CPLAINTEXT_GETWORDINFO, "ECharPosBeyondDocument" );
sl@0
   131
	    }
sl@0
   132
	__ASSERT_DEBUG(aLength+aStartPos<=(DocumentLength()+1),Panic(ECharPosBeyondDocument));
sl@0
   133
sl@0
   134
	__TEST_INVARIANT;
sl@0
   135
	}
sl@0
   136
sl@0
   137
EXPORT_C TInt CPlainText::ToParagraphStart(TInt& aPos) const
sl@0
   138
/** Gets the document position of the start of the paragraph containing the 
sl@0
   139
specified document position.
sl@0
   140
sl@0
   141
@param aPos A document position. Must be valid or a panic occurs. On return, 
sl@0
   142
contains the document position of the first character in the paragraph in 
sl@0
   143
which it is located.
sl@0
   144
@return The number of characters skipped in scanning to the start of the 
sl@0
   145
paragraph. */
sl@0
   146
	{
sl@0
   147
	__TEST_INVARIANT;
sl@0
   148
	if (aPos<0 || aPos>DocumentLength())
sl@0
   149
	    {
sl@0
   150
	    OstTrace0( TRACE_FATAL, CPLAINTEXT_TOPARAGRAPHSTART, "ECharPosBeyondDocument" );
sl@0
   151
	    }
sl@0
   152
	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
sl@0
   153
sl@0
   154
	int skipped = 0;
sl@0
   155
	while (aPos > 0)
sl@0
   156
		{
sl@0
   157
		TPtrC8 p = iByteStore->BackPtr(aPos * sizeof(TText));
sl@0
   158
		const TText* start = (TText*)p.Ptr();
sl@0
   159
		const TText* end = start + p.Length() / sizeof(TText);
sl@0
   160
		while (end > start)
sl@0
   161
			{
sl@0
   162
			end--;
sl@0
   163
			if (*end == EParagraphDelimiter)
sl@0
   164
				return skipped;
sl@0
   165
			aPos--;
sl@0
   166
			skipped++;
sl@0
   167
			}
sl@0
   168
		}
sl@0
   169
	return skipped;
sl@0
   170
	}
sl@0
   171
sl@0
   172
EXPORT_C TInt CPlainText::WordCount()const
sl@0
   173
/** Gets a count of the number of words in the text object.
sl@0
   174
sl@0
   175
@return The number of words in the text object. */
sl@0
   176
	{return CountUnits(EUnitIsWord);}
sl@0
   177
sl@0
   178
EXPORT_C TInt CPlainText::ParagraphCount()const
sl@0
   179
/** Gets a count of the number of paragraphs in the text object.
sl@0
   180
sl@0
   181
Note: the paragraph delimiter which terminates every text object means this 
sl@0
   182
function always returns a count of at least one.
sl@0
   183
sl@0
   184
@return The number of paragraphs contained in the text object. */
sl@0
   185
	{//return CountUnits(EUnitIsParagraph);}
sl@0
   186
	TInt pos=0,unitCount=0;
sl@0
   187
	TUint scanMask=(EScanToUnitStart);
sl@0
   188
	while (pos!=EScanEndOfData)
sl@0
   189
		{
sl@0
   190
		unitCount++;
sl@0
   191
		ScanParas(pos,scanMask);
sl@0
   192
		}
sl@0
   193
	return unitCount;
sl@0
   194
	}
sl@0
   195
sl@0
   196
TInt CPlainText::CountUnits(TUnitOfText aContext)const
sl@0
   197
// Returns the number of units in the document, where
sl@0
   198
// units are defined by the parameter aContext.
sl@0
   199
//
sl@0
   200
	{
sl@0
   201
	SScanData scanData;
sl@0
   202
	TInt pos=0,unitCount=0;
sl@0
   203
	TUint scanMask=(EScanToUnitStart|EScanJoinDelimiters|EScanStayIfBoundary);
sl@0
   204
	InitScanControl(pos,scanMask,aContext,scanData);
sl@0
   205
	ScanUnit(pos,aContext,scanData);
sl@0
   206
	// The above code skips over leading white space
sl@0
   207
	scanData.scanMask &=~ EScanStayIfBoundary;
sl@0
   208
	while(pos!=EScanEndOfData)
sl@0
   209
		{
sl@0
   210
		unitCount++;
sl@0
   211
		ScanUnit(pos,aContext,scanData);
sl@0
   212
		}//	Count complete, so return pos to the beginning of the data
sl@0
   213
	pos=0;
sl@0
   214
	return unitCount;
sl@0
   215
	}
sl@0
   216
sl@0
   217
sl@0
   218
sl@0
   219
sl@0
   220
EXPORT_C TInt CPlainText::ScanWords(TInt& aPos,TUint& aScanMask)const
sl@0
   221
/** Scans from a document position to the beginning or end of a word. The 
sl@0
   222
destination is determined by a scan mask. The scan can either be forwards 
sl@0
   223
(the default) or backwards, and the destination may be the first or last 
sl@0
   224
character in the word containing the position, or the first character in 
sl@0
   225
the next word.
sl@0
   226
sl@0
   227
Note: If an attempt is made to scan beyond the end of text delimiter, on return, 
sl@0
   228
aPos is set to EScanEndOfData  and the function's return value indicates 
sl@0
   229
the number of characters skipped in passing the end of text delimiter.
sl@0
   230
sl@0
   231
@param aPos The document position from which to scan. Must be valid, or a 
sl@0
   232
panic occurs. On return, contains the new document position. 
sl@0
   233
@param aScanMask The scan mask. See the enumeration whose values begin with 
sl@0
   234
CPlainText::EScanBackwards. 
sl@0
   235
@return The number of characters skipped to reach the new document position. */
sl@0
   236
	{
sl@0
   237
	if (aPos<0 || aPos>DocumentLength())
sl@0
   238
	    {
sl@0
   239
	    OstTrace0( TRACE_FATAL, CPLAINTEXT_SCANWORDS, "ECharPosBeyondDocument" );
sl@0
   240
	    }
sl@0
   241
	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
sl@0
   242
	SScanData scanData;
sl@0
   243
	InitScanControl(aPos,aScanMask,EUnitIsWord,scanData);
sl@0
   244
	return ScanUnit(aPos,EUnitIsWord,scanData);
sl@0
   245
	}
sl@0
   246
sl@0
   247
EXPORT_C TInt CPlainText::ScanParas(TInt& aPos,TUint& aScanMask)const
sl@0
   248
/** Scans from a document position to the beginning or end of a paragraph. The 
sl@0
   249
destination is determined by a scan mask. The scan can either be forwards 
sl@0
   250
(the default) or backwards, and the destination may be the first or last 
sl@0
   251
character in the paragraph containing the position, or the first character in 
sl@0
   252
the next paragraph. 
sl@0
   253
sl@0
   254
Note: if an attempt is made to scan beyond the end of text delimiter, on return, 
sl@0
   255
aPos is set to EScanEndOfData ) and the function's return value indicates 
sl@0
   256
the number of characters skipped in passing the end of text delimiter.
sl@0
   257
sl@0
   258
@param aPos The document position to scan from. Must be valid or a panic occurs. 
sl@0
   259
On return, contains the new document position. 
sl@0
   260
@param aScanMask The scan mask. See the enumeration whose values begin with 
sl@0
   261
CPlainText::EScanBackwards. 
sl@0
   262
@return The number of characters skipped to reach the new document position. */
sl@0
   263
	{
sl@0
   264
	if (aPos<0 || aPos>DocumentLength())
sl@0
   265
	    {
sl@0
   266
	    OstTrace0( TRACE_FATAL, CPLAINTEXT_SCANPARAS, "ECharPosBeyondDocument" );
sl@0
   267
	    }
sl@0
   268
	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
sl@0
   269
	SScanData scanData;
sl@0
   270
	InitScanControl(aPos,aScanMask,EUnitIsParagraph,scanData);
sl@0
   271
	return ScanUnit(aPos,EUnitIsParagraph,scanData);
sl@0
   272
	}
sl@0
   273
sl@0
   274
sl@0
   275
void CPlainText::InitScanControl(TInt& aPos,TUint& aScanMask,TUnitOfText aContext,SScanData& aScanData)const
sl@0
   276
// Initialises the control segment of the bitmask
sl@0
   277
// used to determine when scan has completed.
sl@0
   278
// Calls the initialisation of aScanData.
sl@0
   279
//
sl@0
   280
	{
sl@0
   281
	aScanData.pos=aPos;
sl@0
   282
	aScanData.scanMask=0;
sl@0
   283
	aScanData.scanMask|=aScanMask;
sl@0
   284
	if (aScanData.scanMask & EScanStayIfBoundary)
sl@0
   285
		{//Scan one unit the other way first.
sl@0
   286
			aScanData.scanMask ^= EScanBackwards;  // Alter scan direction.
sl@0
   287
			InitScanData(aScanData.pos,aScanData);				
sl@0
   288
			TestForDelimiter(aContext,aScanData);
sl@0
   289
			aScanData.scanMask ^= EScanBackwards;  // Restore original scan direction.
sl@0
   290
		}
sl@0
   291
	InitScanData(aScanData.pos,aScanData);
sl@0
   292
	if(!aPos && (aScanData.scanMask&(EScanStayIfBoundary|EScanBackwards))==EScanStayIfBoundary)
sl@0
   293
		{//At the start of the buffer and scanning forwards and ScanStayIfBoundary set.
sl@0
   294
		aScanData.scanMask &=~aScanData.EInsideUnit;
sl@0
   295
		}
sl@0
   296
	else 
sl@0
   297
		{
sl@0
   298
		TestForDelimiter(aContext,aScanData);
sl@0
   299
		if (aScanData.scanMask & aScanData.EIsDelimiter)
sl@0
   300
			aScanData.scanMask &=~ aScanData.EInsideUnit;
sl@0
   301
		else
sl@0
   302
 			aScanData.scanMask |= aScanData.EInsideUnit;
sl@0
   303
		}
sl@0
   304
	if (aScanData.scanMask & EScanBackwards)
sl@0
   305
		{//Set stop flags relative to scan direction
sl@0
   306
		if (aScanData.scanMask &  EScanToUnitStart)
sl@0
   307
			aScanData.scanMask |= aScanData.EStopEnd;
sl@0
   308
		if (aScanData.scanMask &  EScanToUnitEnd)
sl@0
   309
			aScanData.scanMask |= aScanData.EStopBegin;
sl@0
   310
		}
sl@0
   311
	else
sl@0
   312
		{//Set stop flags relative to scan direction
sl@0
   313
		if (aScanData.scanMask &  EScanToUnitStart)
sl@0
   314
			aScanData.scanMask |= aScanData.EStopBegin;
sl@0
   315
		if (aScanData.scanMask &  EScanToUnitEnd)
sl@0
   316
			aScanData.scanMask |= aScanData.EStopEnd;
sl@0
   317
		}
sl@0
   318
	}
sl@0
   319
sl@0
   320
sl@0
   321
void CPlainText::InitScanData(TInt aPos,SScanData& aScanData)const
sl@0
   322
// Initialises the necessary elements of the scan structure, for ScanXxx 
sl@0
   323
// methods.  These are used to track scan progress.
sl@0
   324
//
sl@0
   325
	{
sl@0
   326
	aScanData.pos=aPos;
sl@0
   327
	aScanData.oldPos=aPos;
sl@0
   328
	aScanData.currentSegLen=0;
sl@0
   329
	if (aScanData.scanMask & EScanBackwards)
sl@0
   330
		{//Scanning backwards.
sl@0
   331
		aScanData.delta=-1;
sl@0
   332
		aScanData.totalBufLen=aPos;
sl@0
   333
		}
sl@0
   334
	else
sl@0
   335
		{//Scanning forwards.
sl@0
   336
		aScanData.delta=+1;
sl@0
   337
		aScanData.totalBufLen=((iByteStore->Size()/sizeof(TText))-aScanData.pos);
sl@0
   338
		}
sl@0
   339
	}	
sl@0
   340
sl@0
   341
sl@0
   342
TInt CPlainText::ScanUnit(TInt& aPos,TUnitOfText aContext,SScanData& aScanData)const
sl@0
   343
// Scan from position aPos, by one unit.
sl@0
   344
// Update aPos to the current position, and return the number
sl@0
   345
// of characters skipped or EScanEndOfData if at the end of the buffer.
sl@0
   346
//
sl@0
   347
	{
sl@0
   348
	while (TestForDelimiter(aContext,aScanData))
sl@0
   349
		{//More data has been read
sl@0
   350
		if (aScanData.scanMask & aScanData.EInsideUnit)
sl@0
   351
			{
sl@0
   352
			if (aScanData.scanMask & aScanData.EIsDelimiter)
sl@0
   353
				{
sl@0
   354
				aScanData.scanMask &=~aScanData.EInsideUnit;
sl@0
   355
				if (aScanData.scanMask & aScanData.EStopEnd)
sl@0
   356
					break;
sl@0
   357
				}
sl@0
   358
			 }
sl@0
   359
		else
sl@0
   360
			{
sl@0
   361
			if (aScanData.scanMask & aScanData.EIsDelimiter)
sl@0
   362
				{
sl@0
   363
				if (!(aScanData.scanMask & EScanJoinDelimiters))
sl@0
   364
					break;
sl@0
   365
				}
sl@0
   366
	   		else
sl@0
   367
				{
sl@0
   368
				aScanData.scanMask |= aScanData.EInsideUnit;
sl@0
   369
				if (aScanData.scanMask & aScanData.EStopBegin)
sl@0
   370
					break;
sl@0
   371
				}
sl@0
   372
			}
sl@0
   373
		}
sl@0
   374
	TInt charsSkipped=(aScanData.scanMask & EScanBackwards)? aPos-aScanData.oldPos : 
sl@0
   375
										aScanData.oldPos-aPos;
sl@0
   376
	aPos=(aScanData.oldPos<(TInt)(iByteStore->Size()/sizeof(TText)))? aScanData.oldPos : EScanEndOfData;
sl@0
   377
	return charsSkipped;
sl@0
   378
	}
sl@0
   379
sl@0
   380
sl@0
   381
TBool CPlainText::TestForDelimiter(TUnitOfText aContext,SScanData& aScanData) const
sl@0
   382
/** Return ETrue if a character was read successfully.
sl@0
   383
Set EIsDelimiter flag if the character read
sl@0
   384
is a delimiter of the appropriate type.*/	
sl@0
   385
    {
sl@0
   386
	aScanData.scanMask |= aScanData.EIsDelimiter;
sl@0
   387
	aScanData.oldPos = aScanData.pos;
sl@0
   388
	TChar c;
sl@0
   389
	if (!GetChar(aScanData,c))
sl@0
   390
		return FALSE;
sl@0
   391
sl@0
   392
	TBool found = FALSE;
sl@0
   393
	if (aContext == EUnitIsParagraph)
sl@0
   394
		found = c == EParagraphDelimiter;
sl@0
   395
	else if (aContext == EUnitIsWord)
sl@0
   396
		{
sl@0
   397
		if (c == EParagraphDelimiter ||
sl@0
   398
			c == ELineBreak ||
sl@0
   399
			c == EPageBreak ||
sl@0
   400
			c == ESpace ||
sl@0
   401
			c == ETabCharacter)
sl@0
   402
			found = TRUE;
sl@0
   403
		if (!found && (aScanData.scanMask & EScanPictureIsDelimiter))
sl@0
   404
			{
sl@0
   405
			if (c == EPictureCharacter)
sl@0
   406
				found = TRUE;
sl@0
   407
			}
sl@0
   408
		if (!found)
sl@0
   409
			{
sl@0
   410
			TChar::TCategory cat = c.GetCategory();
sl@0
   411
			if (cat == TChar::EZsCategory) // it's a space
sl@0
   412
				found = TRUE;
sl@0
   413
			if (!found &&
sl@0
   414
				(aScanData.scanMask & EScanPunctuationIsDelimiter) &&
sl@0
   415
				(cat & TChar::EPunctuationGroup))
sl@0
   416
				{
sl@0
   417
				if (c != EHyphenMinus &&
sl@0
   418
					c != EApostrophe &&
sl@0
   419
					c != EPotentialHyphen &&
sl@0
   420
					c != EHyphen && 
sl@0
   421
					c != ERightSingleQuote)
sl@0
   422
					found = TRUE;
sl@0
   423
				}
sl@0
   424
			}
sl@0
   425
		}
sl@0
   426
sl@0
   427
	if (found)
sl@0
   428
		aScanData.scanMask |= aScanData.EIsDelimiter;
sl@0
   429
	else
sl@0
   430
		aScanData.scanMask &= ~aScanData.EIsDelimiter;
sl@0
   431
	return TRUE;
sl@0
   432
	}
sl@0
   433
sl@0
   434
sl@0
   435
TBool CPlainText::GetChar(SScanData& aScanData,TChar& aChar)const
sl@0
   436
// If there is one, write the next character to aChar, and return ETrue,
sl@0
   437
// otherwise return EFalse.
sl@0
   438
//
sl@0
   439
	{
sl@0
   440
	TPtrC view(_S("a"));  // Dummy value cos no default constructor.
sl@0
   441
	if (!aScanData.currentSegLen)	
sl@0
   442
		{//First time in, or new segment.
sl@0
   443
		if (!aScanData.totalBufLen)
sl@0
   444
			{//No data left to read.
sl@0
   445
			return EFalse;	
sl@0
   446
			}
sl@0
   447
		if (aScanData.delta < 0)
sl@0
   448
			{//Scanning backwards
sl@0
   449
			TPtrC8 tempView=iByteStore->BackPtr(aScanData.pos*sizeof(TText));
sl@0
   450
			view.Set((TText*)tempView.Ptr(),tempView.Length()/sizeof(TText));
sl@0
   451
			}
sl@0
   452
		else
sl@0
   453
			{//Scanning forwards
sl@0
   454
			TPtrC8 tempView=iByteStore->Ptr(aScanData.pos*sizeof(TText));
sl@0
   455
			view.Set((TText*)tempView.Ptr(),tempView.Length()/sizeof(TText));
sl@0
   456
			}//Ptr now returns a TDes.
sl@0
   457
		aScanData.currentSegLen=(TInt)Min(aScanData.totalBufLen,(TInt)view.Length());
sl@0
   458
		aScanData.totalBufLen-=aScanData.currentSegLen;
sl@0
   459
		aScanData.buf=(TText*)view.Ptr();
sl@0
   460
		if (aScanData.delta < 0)
sl@0
   461
			aScanData.buf+=aScanData.currentSegLen-1;
sl@0
   462
		}
sl@0
   463
	aChar=*(aScanData.buf);  // aChar takes the character just read.
sl@0
   464
	aScanData.buf+=aScanData.delta;  // pText increments correctly for both ASCII and Unicode
sl@0
   465
	aScanData.pos+=aScanData.delta;
sl@0
   466
	aScanData.currentSegLen-=1;  // len holds the number of chars left, for ASCII and Unicode
sl@0
   467
	return ETrue;
sl@0
   468
	}
sl@0
   469
sl@0
   470