os/textandloc/textrendering/texthandling/stext/Txtparse.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 "TXTRICH.H"
    20 #include "TXTSTD.H"
    21 #include "ParseLst.h"
    22 #include "OstTraceDefinitions.h"
    23 #ifdef OST_TRACE_COMPILER_IN_USE
    24 #include "TxtparseTraces.h"
    25 #endif
    26 
    27 
    28 
    29 // Install and activate a particular parser, app provides instance
    30 EXPORT_C void CRichText::ActivateParserL(MParser* aParser)
    31 	{
    32 	CParserList* activeParserList = (CParserList*)Dll::Tls();
    33 	if (!activeParserList)
    34 		{
    35 		CreateParserETextTLSL();
    36 		activeParserList = (CParserList*)Dll::Tls();
    37 		}
    38 	activeParserList->ActivateParserL(aParser);
    39 	}
    40 
    41 
    42 // Deactivate and deinstall a particular parser, identified by ptr to instance
    43 EXPORT_C void CRichText::DeactivateParser(MParser* aParser)
    44 	{
    45 	CParserList* activeParserList = (CParserList*)Dll::Tls();
    46 	if (!activeParserList)
    47 	    {
    48 	    OstTrace0( TRACE_DUMP, CRICHTEXT_DEACTIVATEPARSER, "EParserListNotInitialized" );
    49 	    }
    50 	__ASSERT_DEBUG(activeParserList, Panic(EParserListNotInitialized));
    51 	activeParserList->DeactivateParser(aParser);
    52 	if ((activeParserList->iRefCount == 0) && (activeParserList->iNumberInList == 0))
    53 		{
    54 		Dll::FreeTls();
    55 		delete activeParserList;
    56 		}
    57 	}
    58 
    59 
    60 // Install and activate a parser in the default set
    61 EXPORT_C void CRichText::ActivateDefaultParserL(MParser* aParser)
    62 	{
    63 	CParserList* activeParserList = (CParserList*)Dll::Tls();
    64 	if (!activeParserList)
    65 		{
    66 		CreateParserETextTLSL();
    67 		activeParserList = (CParserList*)Dll::Tls();
    68 		}
    69 	activeParserList->ActivateDefaultParserL(aParser);
    70 	}
    71 
    72 
    73 // Deactivate and deinstall the standard set of default parsers
    74 EXPORT_C void CRichText::DeactivateParserDefaults()
    75 	{
    76 	CParserList* activeParserList = (CParserList*)Dll::Tls();
    77 	if (activeParserList)
    78 		{
    79 		activeParserList->DeactivateParserDefaults();
    80 		if ((activeParserList->iRefCount == 0) && (activeParserList->iNumberInList == 0))
    81 			{
    82 			Dll::FreeTls();
    83 			delete activeParserList;
    84 			}
    85 		}
    86 	}
    87 
    88 
    89 // Create ParserLst instance and retain ownership of it but pass address to EText TLS
    90 void CRichText::CreateParserETextTLSL()
    91 	{
    92 	if (Dll::Tls() != NULL)
    93 	    {
    94 	    OstTrace0( TRACE_DUMP, CRICHTEXT_CREATEPARSERETEXTTLSL, "EParserListAlreadyExists" );
    95 	    }
    96 	__ASSERT_DEBUG(Dll::Tls() == NULL, Panic(EParserListAlreadyExists));
    97 	CParserList* activeParserList = new (ELeave) CParserList;
    98 	CleanupStack::PushL(activeParserList);
    99 	TInt err = Dll::SetTls(activeParserList);
   100 	User::LeaveIfError(err);
   101 	CleanupStack::Pop(activeParserList);
   102 	}
   103 
   104 
   105 // Set observer callback to tell whenever the object has been edited.
   106 // If set set then the parser system is active for this instance, otherwise not.
   107 
   108  
   109 
   110 EXPORT_C void CRichText::SetEditObserver(MEditObserver* aEditObserver)
   111 /** Sets the rich text object's edit observer. The observer's EditObserver() function 
   112 is called by the rich text object each time the object's text content is edited 
   113 (e.g. after a paste, insert, delete, reset etc.).
   114 
   115 @param aEditObserver Pointer to the edit observer. */
   116 	{
   117 	iParserData->iEditObserver = aEditObserver;
   118 	}
   119 
   120 	
   121 EXPORT_C TBool CRichText::ParseText(TInt& aStartOfTags, TInt& aLength, TBool aForceScanAllText)
   122 	{
   123 	if (!iIndex.IsPtr())
   124 	    {
   125 	    OstTrace0( TRACE_DUMP, CRICHTEXT_PARSETEXT, "EParserListTextIndexNotInitialized" );
   126 	    }
   127 	__ASSERT_ALWAYS(iIndex.IsPtr(),Panic(EParserListTextIndexNotInitialized));
   128 	TBool foundSomething = EFalse;
   129 	if (iParserData->iActiveParserList && iParserData->iEditObserver)
   130 		{
   131 		if (aForceScanAllText)
   132 			foundSomething = iParserData->iActiveParserList->ParseThisText(*this,0,DocumentLength(),
   133 																		   aStartOfTags,aLength);
   134 		else if (iParserData->HaveRange())
   135 			foundSomething = iParserData->iActiveParserList->ParseThisText(*this,iParserData->StartParse(),
   136 									iParserData->EndParse() - iParserData->StartParse(),
   137 									aStartOfTags,aLength);
   138 		// All parsers have scanned the area
   139 		iParserData->KillRange();
   140 		}
   141 	return foundSomething;
   142 	}
   143 
   144 
   145 // Given a cursor position, is there a tag under it and, if so, give me details	
   146 TBool CRichText::DoCursorOverTag(TInt aPos, MParser*& aParser, TInt& aTagStart, TInt& aLength) const
   147 	{
   148 	TCharFormatX format;
   149 	TCharFormatXMask varies;
   150 	TBool success = EFalse;
   151 	TBuf<1> buf;
   152 
   153 	if (!iParserData->iActiveParserList)
   154 	    {
   155 	    OstTrace0( TRACE_DUMP, CRICHTEXT_DOCURSOROVERTAG, "EParserListNotInitialized" );
   156 	    }
   157 	__ASSERT_DEBUG(iParserData->iActiveParserList, Panic(EParserListNotInitialized));
   158 	if (!iParserData->iEditObserver)
   159 	    {
   160 	    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXT_DOCURSOROVERTAG, "EParserListNotActive" );
   161 	    }
   162 	__ASSERT_DEBUG(iParserData->iEditObserver, Panic(EParserListNotActive));
   163 	GetExtendedCharFormat(format, varies, aPos, 1);
   164 	Extract(buf, aPos, 1);
   165 	if ((format.iParserTag) && (buf[0] != 0x2029))
   166 		{
   167 		aParser = iParserData->iActiveParserList->ParserWithThisTag(format.iParserTag);
   168 		if (aParser == NULL)
   169 			{ // Parser has been deactivated
   170 			return EFalse;
   171 			}
   172 		// Get extent of tag (or set of contiguous tags for same parser)
   173 		TInt pos = aPos;
   174 		TInt startScan;
   175 		TInt scanLength;
   176 		// need to check backwards
   177 		TUint tag = format.iParserTag;	// Stash tag before overwriting it
   178 		for (; pos > 0; pos--)
   179 			{
   180 			GetExtendedCharFormat(format, varies, pos - 1, 1);
   181 			Extract(buf, aPos, 1);
   182 			if ((format.iParserTag != tag) || (buf[0] == 0x2029))
   183 				break;
   184 			}
   185 		startScan = pos;
   186 		// Now forwards
   187 		TInt len = DocumentLength();
   188 		while (pos < len)
   189 			{
   190 			TPtrC ptr;
   191 			GetTextAndExtendedFormat(ptr, format, pos);
   192 			if (format.iParserTag != tag)
   193 				break;
   194 			pos += ptr.Length();
   195 			}
   196 		if (pos > len)
   197 			pos = len;
   198 		scanLength = pos - startScan;
   199 		// Now use the parser that found it originally to isolate the exact range
   200 		// of the tag from the range that could contain several
   201 		for (;;)
   202 			{
   203 			TInt result = aParser->ParseThisText(*this, EFalse, startScan, scanLength, aTagStart, aLength);
   204 			// Check we haven't gone past it (failed to find it this time)
   205 			if (!result || (aTagStart > aPos))
   206 				break;
   207 			if ((aPos >= aTagStart) && (aPos < aTagStart + aLength))
   208 				{
   209 				// We've found it
   210 				success = ETrue;
   211 				break;
   212 				}
   213 			// Not yet, skip over that one
   214 			startScan += aLength;
   215 			scanLength -= aLength;
   216 			if (scanLength < 0)
   217 				break;
   218 			}
   219 	
   220 		}
   221 	return success;
   222 	}
   223 
   224 
   225 EXPORT_C TBool CRichText::CursorOverTag(TInt aPos, MParser*& aParser, TInt& aTagStart, TInt& aLength) const
   226 	{
   227 	TBool over = EFalse;
   228 
   229 	if (iParserData->iActiveParserList && iParserData->iEditObserver)
   230 		{
   231 		iParserData->iLastKnownCursor = aPos;
   232 		if (DoCursorOverTag(aPos, aParser, aTagStart, aLength))
   233 			{
   234 			over = ETrue;
   235 			}
   236 		if (aPos && DoCursorOverTag(aPos - 1, aParser, aTagStart, aLength))
   237 			{
   238 			over = ETrue;
   239 			}
   240 		}
   241 	return over && 
   242 		aParser->ConfirmCursorOverTag(*this, aTagStart, aLength, aPos);
   243 	}
   244 
   245 
   246 EXPORT_C TInt CRichText::PositionOfNextTag(TInt aPos, const MParser * aParser) const
   247 	{
   248 	if (iParserData->iActiveParserList && iParserData->iEditObserver)
   249 		{
   250 		MParser* parser;
   251 		TInt tagStart;
   252 		TInt length;
   253 		TInt newPos = aPos;
   254 		TUint tag = 0;
   255 		if (aParser)
   256 			tag = iParserData->iActiveParserList->TagForThisParser(aParser);
   257 		if (DoCursorOverTag(newPos, parser, tagStart, length))
   258 			newPos = tagStart + length;	// To get past the current tag
   259 		TPtrC ptr;
   260 		TCharFormatX format;
   261 		while (newPos < DocumentLength())
   262 			{
   263 			GetTextAndExtendedFormat(ptr, format, newPos);
   264 			if (format.iParserTag &&(!aParser || format.iParserTag == tag))
   265 				return newPos;
   266 			newPos += ptr.Length();
   267 			}
   268 		}
   269 
   270 	return KErrNotFound;
   271 	}
   272 
   273 
   274 EXPORT_C TInt CRichText::PositionOfNextTag(TInt aPos) const
   275 	{
   276 	return PositionOfNextTag(aPos, NULL);
   277 	}
   278 
   279 
   280 EXPORT_C TInt CRichText::PositionOfPrevTag(TInt aPos, const MParser * aParser) const
   281 	{
   282 	if (iParserData->iActiveParserList && iParserData->iEditObserver)
   283 		{
   284 		MParser* parser;
   285 		TInt tagStart;
   286 		TInt length;
   287 		TInt newPos = aPos;
   288 		TUint tag = 0;
   289 		if (aParser)
   290 			tag = iParserData->iActiveParserList->TagForThisParser(aParser);
   291 		if (DoCursorOverTag(newPos, parser, tagStart, length))
   292 			newPos = tagStart;	// To get past the current tag
   293 		TCharFormatX format;
   294 		TCharFormatXMask varies;
   295 		
   296 		for (; newPos > 0; newPos--)
   297 			{
   298 			GetExtendedCharFormat(format, varies, newPos - 1, 1);
   299 			if (format.iParserTag &&(!aParser || format.iParserTag == tag))
   300 				{
   301 				if (DoCursorOverTag(newPos - 1, parser, tagStart, length))
   302 					return tagStart;
   303 				}
   304 			}
   305 		}
   306 
   307 	return KErrNotFound;
   308 	}
   309 
   310 
   311 EXPORT_C TInt CRichText::PositionOfPrevTag(TInt aPos) const
   312 	{
   313 	return PositionOfPrevTag(aPos, NULL);
   314 	}
   315 
   316 
   317 void CRichText::OverrideFormatForParsersIfApplicable(TPtrC& aText, TCharFormatX& aFormat, TInt aStartPos) const
   318 	{
   319 	if (aFormat.iParserTag && iParserData->iActiveParserList && iParserData->iEditObserver)
   320 		{
   321 		// Replace format
   322 		TInt start = -1;
   323 		TInt length = 0;
   324 		TInt curPos = iParserData->iLastKnownCursor;
   325 		if (curPos > DocumentLength())
   326 			curPos = DocumentLength();  // This shouldn't be neccesary but it makes it more
   327 										// bulletproof if the calls from outside are made in
   328 										// the wrong order
   329 		if (curPos != -1)
   330 			{
   331 			TCharFormatX format;
   332 			TCharFormatXMask varies;
   333 			MParser* parser;
   334 
   335 			GetExtendedCharFormat(format, varies, curPos, 1);
   336 			// If char at curpos has a tag then cursor is over that tag, get extents of tag
   337 			if (CParserList::ReformatOnRollover(format.iParserTag))
   338 				DoCursorOverTag(curPos, parser, start, length);
   339 			else if (curPos)
   340 				{
   341 				GetExtendedCharFormat(format, varies, curPos - 1, 1);
   342 				// Try the char "before" curpos
   343 				if (CParserList::ReformatOnRollover(format.iParserTag))
   344 					DoCursorOverTag(curPos - 1, parser, start, length);
   345 				}
   346 			}
   347 
   348 		MParser* parser = iParserData->iActiveParserList->ParserWithThisTag(aFormat.iParserTag);
   349 		
   350 		if (length && (aStartPos >= start) && (aStartPos < start + length))
   351 			{
   352 			if (start + length < aStartPos + aText.Length())
   353 				aText.Set(aText.Left(start + length - aStartPos));
   354 			if (parser != NULL)
   355 				{
   356 				// Only accept the rollover format if the parser agrees 
   357 				// with the framework match of a tag
   358 				if (parser->ConfirmCursorOverTag(*this, start, length, curPos))
   359 					parser->GetRolloverFormat(aFormat.iCharFormat);
   360 				else 
   361 					// Reset format to recognised format if parser disagrees
   362 					// with the framework match as the tag is still in view
   363 					// and must be formatted as in the else clause below.
   364 					parser->GetRecogniseFormat(aFormat.iCharFormat);
   365 				}
   366 			}
   367 		else
   368 			{
   369 			if (length && (start > aStartPos))
   370 				aText.Set(aText.Left(start - aStartPos));
   371 			if (parser != NULL)
   372 				parser->GetRecogniseFormat(aFormat.iCharFormat);
   373 			}
   374 		}
   375 	}
   376 
   377 
   378 void CRichText::CallEditObserver(TInt aStart, TInt aExtent) const
   379 	{
   380 	if (iParserData->iEditObserver)
   381 		iParserData->iEditObserver->EditObserver(aStart, aExtent);
   382 	}