os/ossrv/lowlevellibsandfws/apputils/src/StringPoolAPI.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#include <e32std.h>
sl@0
    17
#include "StringPoolImplementation.h"
sl@0
    18
sl@0
    19
sl@0
    20
sl@0
    21
EXPORT_C void RStringBase::Close()
sl@0
    22
/** Closes a string. 
sl@0
    23
sl@0
    24
You must close every string you open. In other words, every call to RStringBase::Copy(), 
sl@0
    25
RStringPool::OpenStringL() and RStringPool::OpenFStringL() must be matched 
sl@0
    26
by a close.
sl@0
    27
sl@0
    28
Strings created through RStringPool::String() or RStringPool::StringF() with 
sl@0
    29
an integer argument need not be closed (but closing is harmless). 
sl@0
    30
sl@0
    31
Strings created through RStringPool::String() or RStringPool::StringF() with 
sl@0
    32
a StringToken or StringTokenF argument must not be closed, as they're just 
sl@0
    33
changing the external representation. */
sl@0
    34
	{
sl@0
    35
	if (iVal)
sl@0
    36
		//harmless for empty strings
sl@0
    37
		iPool.iImplementation->Close(RStringTokenEither(iVal));
sl@0
    38
	// invalidate the string if it's a dinamic one	otherwise do nothing
sl@0
    39
	if(!StringUtils::IsTableEntry(this->iVal))
sl@0
    40
		iVal=NULL;
sl@0
    41
	}
sl@0
    42
sl@0
    43
sl@0
    44
EXPORT_C RStringBase RStringBase::Copy()
sl@0
    45
/** Copies a string.
sl@0
    46
sl@0
    47
Both the original and the copy string must be separately closed.
sl@0
    48
sl@0
    49
@return The string base. */
sl@0
    50
	{
sl@0
    51
	if (iVal)
sl@0
    52
		iPool.iImplementation->IncrementCount(RStringTokenEither(iVal));
sl@0
    53
	return *this;
sl@0
    54
	}
sl@0
    55
sl@0
    56
sl@0
    57
/** Returns the content of the string
sl@0
    58
 */
sl@0
    59
EXPORT_C const TDesC8& RStringBase::DesC() const
sl@0
    60
/** Gets the content of the string.
sl@0
    61
sl@0
    62
@return Descriptor containing the content of the string. */
sl@0
    63
	{
sl@0
    64
	// check for empty string
sl@0
    65
	if (iVal == 0)
sl@0
    66
		return KNullDesC8();
sl@0
    67
	if (StringUtils::IsTableEntry(iVal))
sl@0
    68
		{
sl@0
    69
		TInt index = StringUtils::TableIndex(iVal);
sl@0
    70
		TInt tableUid= StringUtils::TableUid(iVal);
sl@0
    71
		return iPool.iImplementation->TableLookup(index, tableUid);
sl@0
    72
		}
sl@0
    73
	else
sl@0
    74
		return *StringUtils::NodePtr(iVal)->iDes;
sl@0
    75
	}
sl@0
    76
sl@0
    77
sl@0
    78
EXPORT_C TInt RStringBase::Index(const TStringTable& aTable) const
sl@0
    79
/** Gets the enumeration value corresponding to this string.
sl@0
    80
sl@0
    81
@param	aTable String table to look in
sl@0
    82
@return	The string's enumeration value, or -1 if there is no such value 
sl@0
    83
@panic	EStringTableNotFound If the table supplied is not found. This panic is raised in debug builds only, in release mode the behaviour is undefined*/
sl@0
    84
	{
sl@0
    85
	//return an error for empty strings(i.e. with iVal=0 )
sl@0
    86
	if(!this->iVal)
sl@0
    87
	   return KErrNotFound;
sl@0
    88
	TInt tableId=iPool.iImplementation->TableUid(aTable);
sl@0
    89
	__ASSERT_DEBUG(tableId!=KErrNotFound,StringPoolPanic::Panic(StringPoolPanic::EStringTableNotFound));
sl@0
    90
	// First check the iVal is part of the same table
sl@0
    91
	if (StringUtils::IsTableEntry(iVal))
sl@0
    92
	   {		
sl@0
    93
	   if (tableId==StringUtils::TableUid(iVal))
sl@0
    94
		   return StringUtils::TableIndex(iVal);
sl@0
    95
	   }		
sl@0
    96
	// Then check if the iVal is in the reverse duplicate list
sl@0
    97
	TInt index=iPool.iImplementation->FindTableIndexFromFirstVal(iVal, tableId);
sl@0
    98
	return index;
sl@0
    99
	}
sl@0
   100
sl@0
   101
sl@0
   102
EXPORT_C const TStringTable* RStringBase::OriginalTableRef() const
sl@0
   103
/** Gets the table (if any) that first added the current string to the pool.
sl@0
   104
sl@0
   105
Note there multiple tables can contain the same string.
sl@0
   106
sl@0
   107
@return The table or NULL if the string was created dynamically (not from 
sl@0
   108
a table) */
sl@0
   109
	{
sl@0
   110
	if(StringUtils::IsTableEntry(iVal))
sl@0
   111
		return &iPool.iImplementation->TableRef(iVal);
sl@0
   112
	else 
sl@0
   113
		return NULL;
sl@0
   114
	}
sl@0
   115
	
sl@0
   116
/** Implementation of RStringPool class*/
sl@0
   117
sl@0
   118
EXPORT_C void RStringPool::OpenL()
sl@0
   119
/** Creates an initialised string pool with no pre-loaded string tables.
sl@0
   120
sl@0
   121
@leave KErrNoMemory Not enough memory to open the pool */
sl@0
   122
	{
sl@0
   123
	iImplementation = new (ELeave) CStringPoolImplementation;
sl@0
   124
	}
sl@0
   125
sl@0
   126
EXPORT_C void RStringPool::OpenL(const TStringTable& aTable,MStringPoolCloseCallBack& aCallBack)
sl@0
   127
/** Creates an initialised string pool with a pre-loaded string table, and a string-pool-closing 
sl@0
   128
callback.
sl@0
   129
sl@0
   130
@param aTable The pre-loaded string table.
sl@0
   131
@param aCallBack Callback interface that is called when the string pool closes
sl@0
   132
@leave KErrNoMemory Not enough memory to open the pool */
sl@0
   133
	{
sl@0
   134
	if (!iImplementation)
sl@0
   135
		{
sl@0
   136
		iImplementation = new (ELeave) CStringPoolImplementation;
sl@0
   137
		CleanupClosePushL(*this);
sl@0
   138
		iImplementation->AddTableL(aTable);
sl@0
   139
		iImplementation->AddCallBackL(aCallBack);
sl@0
   140
		CleanupStack::Pop(); //this
sl@0
   141
		}
sl@0
   142
	else
sl@0
   143
		{
sl@0
   144
		iImplementation->AddTableL(aTable);
sl@0
   145
		iImplementation->AddCallBackL(aCallBack);
sl@0
   146
		}
sl@0
   147
	}
sl@0
   148
sl@0
   149
EXPORT_C void RStringPool::OpenL(const TStringTable& aTable)
sl@0
   150
/** Creates an initialised string pool with a pre-loaded string table.
sl@0
   151
sl@0
   152
@param aTable The pre-loaded string table.
sl@0
   153
@leave KErrNoMemory Not enough memory to open the pool */
sl@0
   154
	{
sl@0
   155
	if (!iImplementation)
sl@0
   156
		{
sl@0
   157
		iImplementation = new (ELeave) CStringPoolImplementation;
sl@0
   158
		CleanupClosePushL(*this);
sl@0
   159
		iImplementation->AddTableL(aTable);
sl@0
   160
		CleanupStack::Pop(); //this
sl@0
   161
		}
sl@0
   162
	else
sl@0
   163
		{
sl@0
   164
		iImplementation->AddTableL(aTable);
sl@0
   165
		}
sl@0
   166
	}
sl@0
   167
sl@0
   168
sl@0
   169
EXPORT_C void RStringPool::Close()
sl@0
   170
/** Closes the string pool table.
sl@0
   171
sl@0
   172
This invalidates all other handles to the table. */
sl@0
   173
	{
sl@0
   174
	delete iImplementation;
sl@0
   175
	iImplementation = NULL;	 
sl@0
   176
	}
sl@0
   177
sl@0
   178
EXPORT_C RStringF RStringPool::OpenFStringL(const TDesC8& aString) const
sl@0
   179
/** Creates an RStringF using the current string pool.
sl@0
   180
sl@0
   181
The string is opened as case-insensitive. 
sl@0
   182
sl@0
   183
@param aString The value of the string.
sl@0
   184
@leave KErrNoMemory Not enough memory to open the string
sl@0
   185
@return Initialised RStringF object */
sl@0
   186
	{
sl@0
   187
	RStringTokenEither newString = iImplementation->OpenL(aString, ETrue);
sl@0
   188
	RStringF r;
sl@0
   189
	r.iPool = *this;
sl@0
   190
	r.iVal = newString.iVal;
sl@0
   191
	return r;
sl@0
   192
	}
sl@0
   193
sl@0
   194
EXPORT_C RString RStringPool::OpenStringL(const TDesC8& aString) const
sl@0
   195
/** Creates an RString using the current string pool.
sl@0
   196
sl@0
   197
The string is opened as case-sensitive. 
sl@0
   198
sl@0
   199
@param aString The value of the string.
sl@0
   200
@leave KErrNoMemory Not enough memory to open the string
sl@0
   201
@return Initialised RString object */
sl@0
   202
	{
sl@0
   203
	RStringTokenEither newString = iImplementation->OpenL(aString, EFalse);
sl@0
   204
	RString r;
sl@0
   205
	r.iPool = *this;
sl@0
   206
	r.iVal = newString.iVal;
sl@0
   207
	return r;
sl@0
   208
	}
sl@0
   209
sl@0
   210
EXPORT_C RString RStringPool::String(RStringToken aString) const
sl@0
   211
/** Creates an RString from the supplied RStringToken. 
sl@0
   212
sl@0
   213
@param aString The string token
sl@0
   214
@return Initialised RString object */
sl@0
   215
	{
sl@0
   216
	RString r;
sl@0
   217
	r.iPool = *this;
sl@0
   218
	r.iVal = aString.iVal;
sl@0
   219
	return r;
sl@0
   220
	}
sl@0
   221
sl@0
   222
EXPORT_C RString RStringPool::String(TInt aIndex,const TStringTable& aTable) const
sl@0
   223
/** Gets a case-sensitive string specified by a string table enumeration value.
sl@0
   224
sl@0
   225
aIndex is interpreted as an offset into the handle's pre-loaded string table. 
sl@0
   226
sl@0
   227
@param	aIndex The string table enumeration value
sl@0
   228
@param	aTable The string table from which to read the string
sl@0
   229
@return	Initialised RString object
sl@0
   230
@panic	EStringTableNotFound If the table supplied is not found. This panic is raised in debug builds only, in release mode the behaviour is undefined*/
sl@0
   231
	{	
sl@0
   232
	__ASSERT_DEBUG(aTable.iCaseSensitive==1,StringPoolPanic::Panic(StringPoolPanic::ECreatingStringWithWrongCase));
sl@0
   233
	if(aIndex <(TInt)aTable.iCount)
sl@0
   234
		{//the index is in valid range the index 
sl@0
   235
		RString r;
sl@0
   236
		r.iPool = *this;
sl@0
   237
		TInt16 tableUid = iImplementation->TableUid(aTable);
sl@0
   238
		__ASSERT_DEBUG(tableUid!=KErrNotFound,StringPoolPanic::Panic(StringPoolPanic::EStringTableNotFound));
sl@0
   239
		r.iVal = StringUtils::ValFromIndex(aIndex, tableUid);
sl@0
   240
		TInt originalVal;	
sl@0
   241
		if (KErrNotFound!=(originalVal=iImplementation->FindFirstValFromDuplicate(r.iVal)))
sl@0
   242
			{
sl@0
   243
			r.iVal=originalVal;
sl@0
   244
			}
sl@0
   245
		return r;
sl@0
   246
		}
sl@0
   247
	else // the index is out of range 
sl@0
   248
	return RString();
sl@0
   249
	}
sl@0
   250
sl@0
   251
EXPORT_C RStringF RStringPool::StringF(RStringTokenF aString) const
sl@0
   252
/** Creates a RStringF from the supplied RStringToken. 
sl@0
   253
sl@0
   254
@param aString The value of the string 
sl@0
   255
@return Initialised RStringF object */
sl@0
   256
	{
sl@0
   257
	RStringF r;
sl@0
   258
	r.iPool = *this;
sl@0
   259
	r.iVal = aString.iVal;
sl@0
   260
	return r;
sl@0
   261
	}
sl@0
   262
sl@0
   263
EXPORT_C RStringF RStringPool::StringF(TInt aIndex,const TStringTable& aTable) const
sl@0
   264
/** Gets a case-insensitive string specified by a string table enumeration value.
sl@0
   265
sl@0
   266
Creates an RStringF from a string table enumeration value.
sl@0
   267
sl@0
   268
aIndex is interpreted as an offset into the handle's pre-loaded string table. 
sl@0
   269
sl@0
   270
@param	aIndex The string table enumeration value
sl@0
   271
@param	aTable The string table from which to read the string
sl@0
   272
@return	Initialised RStringF object
sl@0
   273
@panic	EStringTableNotFound If the table supplied is not found. This panic is raised in debug builds only, in release mode the behaviour is undefined*/
sl@0
   274
	{
sl@0
   275
	__ASSERT_DEBUG(aTable.iCaseSensitive==0,StringPoolPanic::Panic(StringPoolPanic::ECreatingStringWithWrongCase));
sl@0
   276
	if(aIndex <(TInt)aTable.iCount)
sl@0
   277
		{//the index is in valid range the index 
sl@0
   278
		RStringF r;
sl@0
   279
		r.iPool = *this;
sl@0
   280
		TInt16 tableUid = iImplementation->TableUid(aTable);
sl@0
   281
		__ASSERT_DEBUG(tableUid!=KErrNotFound,StringPoolPanic::Panic(StringPoolPanic::EStringTableNotFound));
sl@0
   282
		r.iVal = StringUtils::ValFromIndexF(aIndex, tableUid);
sl@0
   283
		TInt originalVal;	
sl@0
   284
		if (KErrNotFound!=(originalVal=iImplementation->FindFirstValFromDuplicate(r.iVal)))
sl@0
   285
			{
sl@0
   286
			r.iVal=originalVal;
sl@0
   287
			}
sl@0
   288
		return r;
sl@0
   289
		}
sl@0
   290
	else // the index is out of range so return an empty string
sl@0
   291
	return RStringF();
sl@0
   292
	}