os/textandloc/textrendering/texthandling/stext/ParseLst.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /*
     2 * Copyright (c) 1999-2009 Nokia Corporation and/or its subsidiary(-ies).
     3 * All rights reserved.
     4 * This component and the accompanying materials are made available
     5 * under the terms of "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".
     8 *
     9 * Initial Contributors:
    10 * Nokia Corporation - initial contribution.
    11 *
    12 * Contributors:
    13 *
    14 * Description: 
    15 *
    16 */
    17 
    18 
    19 #include "TXTSTD.H"
    20 #include "ParseLst.h"
    21 #include "OstTraceDefinitions.h"
    22 #ifdef OST_TRACE_COMPILER_IN_USE
    23 #include "ParseLstTraces.h"
    24 #endif
    25 
    26 
    27 
    28 //++ sort out definitive hash includes
    29 
    30 
    31 CParserList::CParserItem::CParserItem(MParser* aParser, const TUint aTagIndex)
    32 : iParser(aParser),
    33   iTagIndex(aTagIndex)
    34 	{
    35 	}
    36 
    37 
    38 // Panic the process with UikParse as the category.
    39 void CParserList::Panic(TParserListPanic aPanic) const
    40 	{
    41 	OstTraceExt2( TRACE_FATAL, DUP1_CPARSERLIST_PANIC, "CParserList::Panic;this=%x;aPanic=%x", (unsigned int)this, ( TUint )&( aPanic ) );
    42 	_LIT(panicStr, "ParseLst");
    43 	User::Panic(panicStr, aPanic);
    44 	}
    45 
    46 
    47 CParserList::CParserList()
    48 : iParserList(4)
    49 	{
    50 	iHighestIndex = 0;
    51 	iNumberInList = 0;
    52 	}
    53 
    54 
    55 CParserList::~CParserList()
    56 	{
    57 	iParserList.ResetAndDestroy();	// Reset the list and destroy the CParserItems
    58 	iParserList.Close();			// but leave the parsers and free the resources
    59 	}
    60 
    61 
    62 // Add parser to appropriate list
    63 void CParserList::ActivateAParserL(MParser* aParser, const TBool aDefaultParser)
    64 	{
    65 	// Check the index isn't rolling over
    66 	// N.B. This is somewhat unlikely. Since index is effectively a 29-bit
    67 	// uint (I'm using the top 3 bits as flags) this allows 268435456 parsers
    68 	// to be installed. (Assuming we deinstall almost as many, as we went
    69 	// along, to avoid OOM.) At 50 per sec that's 20 years continuous action!
    70 	if (iHighestIndex >= EMaxParserIndex)
    71 	    {
    72 	    OstTrace0( TRACE_DUMP, CPARSERLIST_ACTIVATEAPARSERL, "EParserIndexRollover" );
    73 	    }
    74 	__ASSERT_DEBUG((iHighestIndex < EMaxParserIndex), Panic(EParserIndexRollover));
    75 	if (!aParser)
    76 	    {
    77 	    OstTrace0( TRACE_DUMP, DUP1_CPARSERLIST_ACTIVATEAPARSERL, "EParserNullPtr" );
    78 	    }
    79 	__ASSERT_DEBUG(aParser, Panic(EParserNullPtr));
    80 	TUint parserIndex = iHighestIndex + 1;
    81 	// Adjust flags to describe parser
    82 	if (aParser->ReformatOnRecognise())
    83 		parserIndex |= static_cast<TUint>( EReformatOnRecogniseFlag );
    84 	if (aParser->ReformatOnRollover())
    85 		parserIndex |= EReformatOnRolloverFlag;
    86 	if (aDefaultParser)
    87 		parserIndex |= EDefaultParserFlag;
    88 	// Create a parser item
    89 	TInt result;
    90 	CParserItem* parserItem = new (ELeave) CParserItem(aParser, parserIndex);
    91 	++iHighestIndex;
    92 	if (aDefaultParser)
    93 		{
    94 		CleanupStack::PushL(parserItem);
    95 		result = iParserList.Append(parserItem);
    96 		if (result) // We couldn't get it in the main list
    97 			User::Leave(result);
    98 		CleanupStack::Pop();
    99 		}
   100 	else
   101 		{
   102 		// Look for the end of the specific parsers and the start of the default set
   103 		TInt count;
   104 		for (count = 0; count < iNumberInList; count++)
   105 			{
   106 			if (DefaultParser(iParserList[count]->TagIndex()))
   107 				break;
   108 			}
   109 		CleanupStack::PushL(parserItem);
   110 		result = iParserList.Insert(parserItem, count);
   111 		if (result) // We couldn't get it in the main list
   112 			User::Leave(result);
   113 		CleanupStack::Pop();
   114 		}
   115 	iNumberInList++;
   116 	}
   117 
   118 
   119 // Activate an individual parser
   120 void CParserList::ActivateParserL(MParser* aParser)
   121 	{
   122 	ActivateAParserL(aParser, EFalse);
   123 	}
   124 
   125 
   126 // N.B. We can't just delete CParserList and zero TLS when we get back
   127 // to a list with no specific or default parsers because there could be
   128 // an EText left with a local ptr to it.
   129 void CParserList::DeactivateParser(MParser* aParser)
   130 	{
   131 	if (!iNumberInList)
   132 	    {
   133 	    OstTrace0( TRACE_DUMP, CPARSERLIST_DEACTIVATEPARSER, "EParserNoneActive" );
   134 	    }
   135 	__ASSERT_DEBUG(iNumberInList, Panic(EParserNoneActive));
   136 	if (!aParser)
   137 	    {
   138 	    OstTrace0( TRACE_DUMP, DUP1_CPARSERLIST_DEACTIVATEPARSER, "EParserNullPtr" );
   139 	    }
   140 	__ASSERT_DEBUG(aParser, Panic(EParserNullPtr));
   141 	// Run thru list till find entry we need
   142 	TInt count;
   143 	for (count = 0; count < iNumberInList; count++)
   144 		{
   145 		if (iParserList[count]->Parser() == aParser)
   146 			{
   147 			delete iParserList[count];
   148 			iParserList.Remove(count);
   149 			break;
   150 			}
   151 		}
   152 	if (count >= iNumberInList)
   153 	    {
   154 	    OstTrace0( TRACE_DUMP, DUP2_CPARSERLIST_DEACTIVATEPARSER, "EParserInstanceNotActive" );
   155 	     }
   156 	__ASSERT_DEBUG((count < iNumberInList), Panic(EParserInstanceNotActive));
   157 	iNumberInList--;
   158 	}
   159 
   160 
   161 // Activate a parser as one of the default set
   162 void CParserList::ActivateDefaultParserL(MParser* aParser)
   163 	{
   164 	ActivateAParserL(aParser, ETrue);
   165 	}
   166 
   167 
   168 //++ Put comment here
   169 void CParserList::DeactivateParserDefaults()
   170 	{
   171 	if (iNumberInList)
   172 		{
   173 		// Take them out of the list
   174 		while (iNumberInList && DefaultParser(iParserList[iNumberInList - 1]->TagIndex()))
   175 			{
   176 			// Tell the parser to free itself
   177 			iParserList[iNumberInList - 1]->Parser()->Release();
   178 			// Delete the item that refers to it
   179 			delete iParserList[iNumberInList - 1];
   180 			// Remove the entry from the list
   181 			iParserList.Remove(iNumberInList - 1);
   182 			iNumberInList--;
   183 			}
   184 		}
   185 	}
   186 
   187 
   188 // Called by EText to scan an area of text
   189 TBool CParserList::ParseThisText(CRichText& aTextObj,TInt aStartScan,TInt aScanLength,TInt& aStartOfTags,TInt& aLength)
   190 	{
   191 	TBool foundSomething = EFalse;
   192 	TCharFormatX format;
   193 	TCharFormatXMask varies;
   194 	TPtrC ptr;
   195 	TInt endRange;
   196 
   197 	// Scan either side of the range in case part of some tagged text was deleted.
   198 	if (aStartScan > 0)
   199 		{
   200 		aStartScan--;
   201 		aScanLength++;
   202 		}
   203 	if (aStartScan + aScanLength < aTextObj.DocumentLength())
   204 		aScanLength++;
   205 
   206 	if (iNumberInList && aScanLength)
   207 		{
   208 		aStartOfTags = aStartScan + aScanLength;
   209 		aLength = 0;
   210 		for (TInt count = 0; count < iNumberInList; count++)
   211 			{
   212 			// For each parser in the list
   213 			TUint tagIndex = iParserList[count]->TagIndex();
   214 			TInt pos;
   215 			MParser* parser = iParserList[count]->Parser();
   216 			TInt localStartScan = aStartScan;
   217 			TInt localScanLength = aScanLength;
   218 			// Start by removing existing tags for this parser. This ensures
   219 			// that old tags that have been invalidated by subsequent editing
   220 			// are removed. Any that are still valid will be replaced.
   221 			aTextObj.GetExtendedCharFormat(format, varies, localStartScan, aScanLength);
   222 
   223 			if (format.iParserTag || varies.AttribIsSet(EAttParserTag))
   224 				{
   225 				// At least some of the object contains a non-zero tag - go through it
   226 				// Are we starting part way through a tag?
   227 				aTextObj.GetExtendedCharFormat(format, varies, localStartScan, 1);
   228 				if (format.iParserTag == tagIndex)
   229 					{
   230 					// The first char of this range has the current parsers tag
   231 					// so we need to check backwards for the start of that tag
   232 					for (pos = localStartScan; pos > 0; pos--)
   233 						{
   234 						aTextObj.GetExtendedCharFormat(format, varies, pos - 1, 1);
   235 						if (format.iParserTag != tagIndex)
   236 							break;
   237 						}
   238 					// Since we are going to remove a tag starting from here
   239 					// we need to allow this area to be rescanned
   240 					localScanLength += localStartScan - pos;
   241 					localStartScan = pos;
   242 					}
   243 				// What about off the end?
   244 				aTextObj.GetExtendedCharFormat(format, varies, localStartScan + localScanLength - 1, 1);
   245 				if (format.iParserTag == tagIndex)
   246 					{
   247 					// The last char of this range has the current parsers tag
   248 					// so we need to check forwards for the end of that tag
   249 					pos = localStartScan + localScanLength;
   250 					TInt end = aTextObj.DocumentLength();
   251 					while (pos < end)
   252 						{
   253 						aTextObj.GetTextAndExtendedFormat(ptr, format, pos);
   254 						if (format.iParserTag != tagIndex)
   255 							break;
   256 						pos += ptr.Length();
   257 						}
   258 					// Adjust scan length
   259 					localScanLength = pos - localStartScan;
   260 					}
   261 				pos = localStartScan;
   262 				while (pos < localStartScan + localScanLength)
   263 					{
   264 					// Run along the scan range
   265 					aTextObj.GetTextAndExtendedFormat(ptr, format, pos);
   266 					if (format.iParserTag == tagIndex)
   267 						{
   268 						// Remove this tag
   269 						format.iParserTag = 0;
   270 						varies.ClearAll();
   271 						varies.SetAttrib(EAttParserTag);
   272 						TRAPD(leaveCode, aTextObj.ApplyExtendedCharFormatL(format, varies, pos, ptr.Length()));
   273 						if (leaveCode==KErrNone)
   274 							foundSomething = ETrue;	// We are removing a tag
   275 						if (aLength)
   276 							{
   277 							if (pos < aStartOfTags)
   278 								{
   279 								aLength += aStartOfTags - pos;
   280 								aStartOfTags = pos;
   281 								}
   282 							if (pos + ptr.Length() > aStartOfTags + aLength)
   283 								aLength = pos + ptr.Length() - aStartOfTags;
   284 							}
   285 						else
   286 							{
   287 							aStartOfTags = pos;
   288 							aLength = ptr.Length();
   289 							}
   290 						}
   291 					pos += ptr.Length();
   292 					}
   293 				}
   294 			endRange = localStartScan + localScanLength;
   295 
   296 			// For this parser, run through text looking for changes
   297 			TBool allowBack = ETrue;
   298 			for (;;)	// Run forever
   299 				{
   300 				TInt localStartTag = aTextObj.DocumentLength();
   301 				TInt localTagLength = 0;
   302 				TInt result = parser->ParseThisText(aTextObj, allowBack, localStartScan, localScanLength, localStartTag, localTagLength);
   303 				if (!result)
   304 					break;
   305 				if (!allowBack && (localStartTag < localStartScan))
   306 				    {
   307 				    OstTrace0( TRACE_DUMP, CPARSERLIST_PARSETHISTEXT, "EParserIgnoringAllowFlag" );
   308 				    }
   309 				__ASSERT_DEBUG(allowBack || (localStartTag >= localStartScan), Panic(EParserIgnoringAllowFlag));
   310 				TInt startNewTag = localStartTag;
   311 				TInt lengthNewTag = localTagLength;
   312 				aTextObj.GetExtendedCharFormat(format, varies, localStartTag, localTagLength);
   313 				if (format.iParserTag || varies.AttribIsSet(EAttParserTag))
   314 					{
   315 					// At least some of this area contains a non-zero tag - go through it
   316 					pos = localStartTag;
   317 					TBool higher = EFalse;
   318 					while (pos < localStartTag + localTagLength)
   319 						{
   320 						aTextObj.GetTextAndExtendedFormat(ptr, format, pos);
   321 						if (format.iParserTag && (MaskedTag(format.iParserTag) < MaskedTag(tagIndex)))
   322 							{
   323 							// A higher precedence tag is already here so we can't
   324 							// insert our tag - let's see how far it goes
   325 							TUint tag = format.iParserTag;	// Stash tag before overwriting it
   326 							TInt len = aTextObj.DocumentLength();
   327 							while (pos < len)
   328 								{
   329 								aTextObj.GetTextAndExtendedFormat(ptr, format, pos);
   330 								if (format.iParserTag != tag)
   331 									break;
   332 								pos += ptr.Length();
   333 								}
   334 							result = EFalse;
   335 							startNewTag = pos;
   336 							lengthNewTag = 0;
   337 							break;
   338 							}
   339 						// If there aren't any higher precedence tags in here then this
   340 						// will save us having to go through again if there aren't any
   341 						// lower precedence tags either
   342 						if (MaskedTag(format.iParserTag) >= MaskedTag(tagIndex))
   343 							higher = ETrue;
   344 						pos += ptr.Length();
   345 						}
   346 					if (higher)
   347 						{
   348 						// There are lower or equal precedence tags in this range
   349 						// Do they extend back off the start?
   350 						aTextObj.GetExtendedCharFormat(format, varies, localStartTag, 1);
   351 						if (format.iParserTag)
   352 							{
   353 							// need to check backwards
   354 							TUint tag = format.iParserTag;	// Stash tag before overwriting it
   355 							for (pos = localStartTag; pos > 0; pos--)
   356 								{
   357 								aTextObj.GetExtendedCharFormat(format, varies, pos - 1, 1);
   358 								if (format.iParserTag != tag)
   359 									break;
   360 								}
   361 							localTagLength += localStartTag - pos;
   362 							localStartTag = pos;
   363 							}
   364 						// What about off the end?
   365 						pos = localStartTag + localTagLength;
   366 						aTextObj.GetExtendedCharFormat(format, varies, pos - 1, 1);
   367 						if (format.iParserTag)
   368 							{
   369 							// need to check forwards
   370 							TUint tag = format.iParserTag;	// Stash tag before overwriting it
   371 							TInt len = aTextObj.DocumentLength();
   372 							while (pos < len)
   373 								{
   374 								aTextObj.GetTextAndExtendedFormat(ptr, format, pos);
   375 								if (format.iParserTag != tag)
   376 									break;
   377 								pos += ptr.Length();
   378 								}
   379 							localTagLength = pos - localStartTag;
   380 							}
   381 
   382 						// Remove all tags in this area - they all have lower precedence
   383 						format.iParserTag = 0;
   384 						varies.ClearAll();
   385 						varies.SetAttrib(EAttCharLanguage);
   386 						TRAPD(leaveCode, aTextObj.ApplyExtendedCharFormatL(format, varies, localStartTag, localTagLength));
   387 						if (leaveCode==KErrNone)
   388 							foundSomething = ETrue;	// We are removing a tag
   389 						}
   390 					}
   391 
   392 				if (result)
   393 					{
   394 					// Format tag this area with tagIndex
   395 					format.iParserTag = tagIndex;
   396 					varies.ClearAll();
   397 					varies.SetAttrib(EAttParserTag);
   398 					// Use the original length, not the possibly expanded version
   399 					TRAPD(leaveCode, aTextObj.ApplyExtendedCharFormatL(format, varies, startNewTag, lengthNewTag));
   400 					if (leaveCode==KErrNone)
   401 						foundSomething = ETrue;		// We are applying a tag
   402 					if (aLength)
   403 						{
   404 						if (localStartTag < aStartOfTags)
   405 							{
   406 							aLength += aStartOfTags - localStartTag;
   407 							aStartOfTags = localStartTag;
   408 							}
   409 						if (localStartTag + localTagLength > aStartOfTags + aLength)
   410 							aLength = localStartTag + localTagLength - aStartOfTags;
   411 						}
   412 					else
   413 						{
   414 						aStartOfTags = localStartTag;
   415 						aLength = localTagLength;
   416 						}
   417 					}
   418 				// Jump over
   419 				localScanLength -= startNewTag + lengthNewTag - localStartScan;
   420 				localStartScan = startNewTag + lengthNewTag;	// Adjust start of next scan run
   421 				if (localStartScan >= endRange)	// Have we reached the end of the range yet?
   422 					break;
   423 				allowBack = EFalse;
   424 				}
   425 			}
   426 		}
   427 
   428 	return foundSomething;
   429 	}
   430 
   431 
   432 // given a tag, fetch a ptr to the parser - or null
   433 MParser* CParserList::ParserWithThisTag(const TUint aTagIndex) const
   434 	{
   435 	MParser* parser = NULL;
   436 	for (TInt count = 0; count < iNumberInList; count++)
   437 		{
   438 		if (aTagIndex == iParserList[count]->TagIndex())
   439 			{
   440 			parser = iParserList[count]->Parser();
   441 			break;
   442 			}
   443 		}
   444 	return parser;
   445 	}
   446 
   447 // given a ptr to a parser, fetch a tag - or zero
   448 TUint CParserList::TagForThisParser(const MParser *const aParser) const
   449 	{
   450 	TUint tagIndex = 0;
   451 	for (TInt count = 0; count < iNumberInList; count++)
   452 		{
   453 		if (aParser == iParserList[count]->Parser())
   454 			{
   455 			tagIndex = iParserList[count]->TagIndex();
   456 			break;
   457 			}
   458 		}
   459 	if (!tagIndex)
   460 	    {
   461 	    OstTrace0( TRACE_DUMP, CPARSERLIST_TAGFORTHISPARSER, "EParserNoSuchTag" );
   462 	    }
   463 	__ASSERT_DEBUG(tagIndex, Panic(EParserNoSuchTag));
   464 	return tagIndex;
   465 	}
   466 
   467 
   468 CParserData::CParserData(TInt aEndParse):
   469 	iStartParse(0),
   470 	iEndParse(aEndParse)
   471 	{
   472 	// Get parser data
   473 	iActiveParserList = (CParserList*)Dll::Tls();
   474 	iLastKnownCursor = -1;
   475 	if (iActiveParserList)
   476 		iActiveParserList->iRefCount++;
   477 	}
   478 
   479 
   480 CParserData::~CParserData()
   481 	{
   482 	if (iActiveParserList)
   483 		{
   484 		iActiveParserList->iRefCount--;
   485 		if ((iActiveParserList->iRefCount == 0) && (iActiveParserList->iNumberInList == 0))
   486 			{
   487 			Dll::FreeTls();
   488 			delete iActiveParserList;
   489 			}
   490 		}
   491 	}
   492 
   493 
   494 // Merge the specified range, which may have changed length, into the current range.
   495 // aOldLength is the # of chars deleted and to be removed from the range
   496 // aNewLength is the # of chars inserted and to be added to the range
   497 void CParserData::MergeRange(TInt aStart,TInt aOldLength,TInt aNewLength)
   498 	{
   499 	if (iStartParse == -1) // no current range
   500 		{
   501 		iStartParse = aStart;
   502 		iEndParse = aStart + aNewLength;
   503 		}
   504 	else
   505 		{
   506 		if (aStart < iStartParse)
   507 			iStartParse = aStart;
   508 		if (aStart + aOldLength > iEndParse)
   509 			iEndParse = aStart + aOldLength;
   510 		iEndParse += aNewLength - aOldLength;
   511 		}
   512 	}