os/ossrv/genericservices/mimerecognitionfw/apmime/APMREC.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) 1997-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
// Defines CApaDataRecognizerType the base class for concrete mime recognizers that
sl@0
    15
// perform the actual mime type recognitionand the CApaDataRecognizer that provides
sl@0
    16
// the interface to the collection of recognizers.
sl@0
    17
// 
sl@0
    18
//
sl@0
    19
sl@0
    20
sl@0
    21
#include "APMSTD.H"
sl@0
    22
#include "APMREC.H"
sl@0
    23
#include "APMPAN.H"
sl@0
    24
#include "APMFNDR.H"
sl@0
    25
sl@0
    26
#ifdef USING_ECOM_RECOGS
sl@0
    27
#include <ecom/ecom.h>
sl@0
    28
#endif
sl@0
    29
sl@0
    30
NONSHARABLE_CLASS(CDataRecognizerExtension) : public CBase
sl@0
    31
	{
sl@0
    32
public:
sl@0
    33
	inline static CDataRecognizerExtension* NewL() {return new(ELeave) CDataRecognizerExtension;}
sl@0
    34
	inline TUid DestructorUid() const {return iDtorKey;}
sl@0
    35
	inline TUid& DestructorUidReference() {return iDtorKey;}
sl@0
    36
	inline CApaDataRecognizer& DataRecognizer() {return *iDataRecognizer;}
sl@0
    37
	inline void SetDataRecognizer(CApaDataRecognizer& aDataRecognizer) {iDataRecognizer=&aDataRecognizer;}
sl@0
    38
private:
sl@0
    39
	inline CDataRecognizerExtension() {}
sl@0
    40
private:
sl@0
    41
	CApaDataRecognizer* iDataRecognizer; // no ownership
sl@0
    42
	TUid iDtorKey; // destructor key to track the instance of ECOM implementation class
sl@0
    43
	};
sl@0
    44
sl@0
    45
class TDataToRecognize
sl@0
    46
	{
sl@0
    47
public:
sl@0
    48
	TDataToRecognize(const TDesC& aName, const TDesC8& aBuffer);
sl@0
    49
	TDataToRecognize(CApaDataRecognizer& aDataRecognizer, RFile& aFile, TInt aPreferredBufSize);
sl@0
    50
	void RecognizeL(CApaDataRecognizerType& aDataRecognizerType);
sl@0
    51
	void PrepareForRecognizeLoopLC();
sl@0
    52
	void PopOffCleanupStackAndResetAfterRecognizeLoopL();
sl@0
    53
private:
sl@0
    54
	enum
sl@0
    55
		{
sl@0
    56
		EFlagFilePositionStored	=0x00000001,
sl@0
    57
		EFlagLeaveIfError		=0x00000002
sl@0
    58
		};
sl@0
    59
private:
sl@0
    60
	static void ResetAfterRecognizeLoopL(TAny* aThis);
sl@0
    61
	void ResetAfterRecognizeLoopL();
sl@0
    62
private:
sl@0
    63
	TUint iFlags;
sl@0
    64
	const TDesC* iName; // owned if iFile!=NULL
sl@0
    65
	const TDesC8* iBuffer; // owned if iFile!=NULL
sl@0
    66
	CApaDataRecognizer* iDataRecognizer;
sl@0
    67
	RFile* iFile;
sl@0
    68
	TInt iFilePosition;
sl@0
    69
	TInt iPreferredBufSize;
sl@0
    70
	};
sl@0
    71
sl@0
    72
//
sl@0
    73
// class CApaDataRecognizer
sl@0
    74
//
sl@0
    75
sl@0
    76
EXPORT_C TDataRecognitionResult CApaDataRecognizer::RecognizeL(const TDesC& aName, const TDesC8& aBuffer)
sl@0
    77
	{
sl@0
    78
	TDataToRecognize dataToRecognize(aName, aBuffer);
sl@0
    79
	return RecognizeL(dataToRecognize);
sl@0
    80
	}
sl@0
    81
sl@0
    82
EXPORT_C TDataRecognitionResult CApaDataRecognizer::RecognizeL(RFile& aFile, TInt aPreferredBufSize)
sl@0
    83
	{
sl@0
    84
	TDataToRecognize dataToRecognize(*this, aFile, aPreferredBufSize);
sl@0
    85
	return RecognizeL(dataToRecognize);
sl@0
    86
	}
sl@0
    87
sl@0
    88
TDataRecognitionResult CApaDataRecognizer::RecognizeL(TDataToRecognize& aDataToRecognize)
sl@0
    89
// attempt to recognize using all recognizers
sl@0
    90
// keeps recognizing until either a recognizer's confidence meets or 
sl@0
    91
// exceeds the currently acceptable confidence, or all recognizers 
sl@0
    92
// have been used
sl@0
    93
	{
sl@0
    94
	iResult.Reset();
sl@0
    95
	aDataToRecognize.PrepareForRecognizeLoopLC();
sl@0
    96
	TInt storedErr=KErrNone;
sl@0
    97
	TInt count=iDataRecognizerList.Count();
sl@0
    98
	for (TInt i=0; i<count; i++)
sl@0
    99
		{
sl@0
   100
		if (DoRecognize(iDataRecognizerList[i], aDataToRecognize, TDataType(), storedErr))
sl@0
   101
			{
sl@0
   102
			break;
sl@0
   103
			}
sl@0
   104
		}
sl@0
   105
	aDataToRecognize.PopOffCleanupStackAndResetAfterRecognizeLoopL();
sl@0
   106
	// if we haven't recognized anything and there was a problem with a recognizer, propagate
sl@0
   107
	// the error.  n.b. if several recognizers had errors, only propagate the first error we 
sl@0
   108
	// encountered
sl@0
   109
	if (iResult.iConfidence==CApaDataRecognizerType::ENotRecognized)
sl@0
   110
		{
sl@0
   111
		User::LeaveIfError(storedErr);
sl@0
   112
		}
sl@0
   113
	return iResult;
sl@0
   114
	}
sl@0
   115
sl@0
   116
EXPORT_C TBool CApaDataRecognizer::RecognizeL(const TDesC& aName, const TDesC8& aBuffer, const TDataType& aDataType)
sl@0
   117
	{
sl@0
   118
	TDataToRecognize dataToRecognize(aName, aBuffer);
sl@0
   119
	return RecognizeL(dataToRecognize, aDataType);
sl@0
   120
	}
sl@0
   121
sl@0
   122
EXPORT_C TBool CApaDataRecognizer::RecognizeL(RFile& aFile, TInt aPreferredBufSize, const TDataType& aDataType)
sl@0
   123
	{
sl@0
   124
	TDataToRecognize dataToRecognize(*this, aFile, aPreferredBufSize);
sl@0
   125
	return RecognizeL(dataToRecognize, aDataType);
sl@0
   126
	}
sl@0
   127
sl@0
   128
TBool CApaDataRecognizer::RecognizeL(TDataToRecognize& aDataToRecognize, const TDataType& aDataType)
sl@0
   129
// recognize using only recognizers which supply the aDataType
sl@0
   130
	{
sl@0
   131
	iResult.Reset();
sl@0
   132
	TInt count=iDataRecognizerList.Count();
sl@0
   133
	TInt storedErr=KErrNone;
sl@0
   134
	aDataToRecognize.PrepareForRecognizeLoopLC();
sl@0
   135
	for (TInt i=0; i<count; i++)
sl@0
   136
		{
sl@0
   137
		CApaDataRecognizerType* rec=iDataRecognizerList[i];
sl@0
   138
		TInt countMime=rec->MimeTypesCount();
sl@0
   139
		for (TInt ii=0; ii<countMime; ii++)
sl@0
   140
			{
sl@0
   141
			TDataType supportedDataType;
sl@0
   142
			TRAPD(err, supportedDataType=rec->SupportedDataTypeL(ii));
sl@0
   143
			if (err == KErrNone)
sl@0
   144
				{
sl@0
   145
				if (supportedDataType == aDataType)
sl@0
   146
					{
sl@0
   147
					if (DoRecognize(rec, aDataToRecognize, aDataType, storedErr))
sl@0
   148
						{
sl@0
   149
						aDataToRecognize.PopOffCleanupStackAndResetAfterRecognizeLoopL();
sl@0
   150
						return ETrue;
sl@0
   151
						}
sl@0
   152
					break;
sl@0
   153
					}
sl@0
   154
				}
sl@0
   155
			else if (storedErr==KErrNone)
sl@0
   156
				{
sl@0
   157
				// this is the first time we've hit an error with SupportedDataType,
sl@0
   158
				// so store it now and propagate later if we don't find a suitable 
sl@0
   159
				// recognizer
sl@0
   160
				storedErr=err;
sl@0
   161
				}
sl@0
   162
			}
sl@0
   163
		}
sl@0
   164
	aDataToRecognize.PopOffCleanupStackAndResetAfterRecognizeLoopL();
sl@0
   165
	if (iResult.iConfidence==CApaDataRecognizerType::ENotRecognized)
sl@0
   166
		{
sl@0
   167
		// a suitable recognizer wasn't found, but we may have encountered an
sl@0
   168
		// error with one of the recognizers we scanned n.b. only the first error
sl@0
   169
		// encountered is propagated
sl@0
   170
		User::LeaveIfError(storedErr);
sl@0
   171
		}
sl@0
   172
	return iResult.iConfidence!=CApaDataRecognizerType::ENotRecognized;
sl@0
   173
	}
sl@0
   174
sl@0
   175
TBool CApaDataRecognizer::DoRecognize(CApaDataRecognizerType* aDataRecognizerType, TDataToRecognize& aDataToRecognize, const TDataType& aDataType, TInt& aError)
sl@0
   176
// do the recognition, if the recognizer is sufficiently confident, return ETrue
sl@0
   177
	{
sl@0
   178
	TDataRecognitionResult result;
sl@0
   179
	TRAPD(err, result=aDataRecognizerType->RecognizeL(aDataToRecognize));
sl@0
   180
	if (err != KErrNone)
sl@0
   181
		{
sl@0
   182
		// We only want to store the first error, it if is not KErrNone, it means
sl@0
   183
		// we have set this error previously
sl@0
   184
		if (aError==KErrNone)
sl@0
   185
			{
sl@0
   186
			aError=err;
sl@0
   187
			}
sl@0
   188
		return EFalse;
sl@0
   189
		}
sl@0
   190
 
sl@0
   191
	if (aDataType!=result.iDataType && aDataType!=TDataType())
sl@0
   192
		{
sl@0
   193
		return EFalse;
sl@0
   194
		}
sl@0
   195
	if (result.iConfidence>iResult.iConfidence)
sl@0
   196
		{
sl@0
   197
		iResult=result;
sl@0
   198
		return (result.iConfidence>=iAcceptedConfidence);
sl@0
   199
		}
sl@0
   200
	return EFalse;
sl@0
   201
	}
sl@0
   202
sl@0
   203
EXPORT_C TInt CApaDataRecognizer::AcceptedConfidence() const
sl@0
   204
	{
sl@0
   205
	return iAcceptedConfidence;
sl@0
   206
	}
sl@0
   207
sl@0
   208
EXPORT_C void CApaDataRecognizer::SetAcceptedConfidence(TInt aConfidence)
sl@0
   209
	{
sl@0
   210
	iAcceptedConfidence=aConfidence;
sl@0
   211
	}
sl@0
   212
sl@0
   213
EXPORT_C CApaDataRecognizer::~CApaDataRecognizer()
sl@0
   214
	{
sl@0
   215
	iDataRecognizerList.ResetAndDestroy();
sl@0
   216
	iDataArray.Reset();
sl@0
   217
	}
sl@0
   218
sl@0
   219
EXPORT_C CApaDataRecognizer::CApaDataRecognizer(RFs& aFs)
sl@0
   220
	:iFs(aFs),
sl@0
   221
	iMaxBufferSize(-1),
sl@0
   222
	iDataRecognizerList(KDataArrayGranularity),
sl@0
   223
	iDataArray(KDataArrayGranularity),
sl@0
   224
	iAcceptedConfidence(CApaDataRecognizerType::ECertain)
sl@0
   225
	{}
sl@0
   226
sl@0
   227
EXPORT_C void CApaDataRecognizer::AddDataRecognizerTypeL(CApaDataRecognizerType* aDataRecognizerType)
sl@0
   228
// add the concrete recognizer into the list at the appropriate priority
sl@0
   229
	{
sl@0
   230
#if defined(_DEBUG)
sl@0
   231
 	CApaDataRecognizerType* rec=NULL;
sl@0
   232
	const TInt countDeb=iDataRecognizerList.Count();
sl@0
   233
	for (TInt ii=0; ii<countDeb; ii++)
sl@0
   234
		{
sl@0
   235
		rec=iDataRecognizerList[ii];
sl@0
   236
 		__ASSERT_ALWAYS(rec->TypeUid()!=aDataRecognizerType->TypeUid(), Panic(EDPanicDuplicateRecognizer));
sl@0
   237
		}
sl@0
   238
#endif // _DEBUG
sl@0
   239
sl@0
   240
	TInt priority=aDataRecognizerType->Priority();
sl@0
   241
	TInt count=iDataRecognizerList.Count();
sl@0
   242
	for (TInt i=0; i<count; i++)
sl@0
   243
		{
sl@0
   244
		if (iDataRecognizerList[i]->Priority()<priority)
sl@0
   245
			{
sl@0
   246
			iDataRecognizerList.InsertL(aDataRecognizerType, i);
sl@0
   247
			aDataRecognizerType->DataRecognizerExtension()->SetDataRecognizer(*this);
sl@0
   248
			return;
sl@0
   249
			}
sl@0
   250
		}
sl@0
   251
	iDataRecognizerList.AppendL(aDataRecognizerType);
sl@0
   252
	aDataRecognizerType->DataRecognizerExtension()->SetDataRecognizer(*this);
sl@0
   253
	}
sl@0
   254
sl@0
   255
EXPORT_C void CApaDataRecognizer::CApaDataRecognizer_Reserved_1()
sl@0
   256
	{}
sl@0
   257
sl@0
   258
EXPORT_C TInt CApaDataRecognizer::RemoveDataRecognizerType(const CApaDataRecognizerType* aDataRecognizerType)
sl@0
   259
// remove the concrete recognizer from the list
sl@0
   260
// if it is not locked, if it fails return an error code
sl@0
   261
	{
sl@0
   262
	// find the recognizer in the list
sl@0
   263
	TInt count=iDataRecognizerList.Count();
sl@0
   264
	TUid uid=aDataRecognizerType->TypeUid();
sl@0
   265
	TInt i;
sl@0
   266
	for (i=0; i<count; i++)
sl@0
   267
		{
sl@0
   268
		if (iDataRecognizerList[i]->TypeUid()==uid)
sl@0
   269
			break;
sl@0
   270
		}
sl@0
   271
	// did we find a match
sl@0
   272
	if (i==count)
sl@0
   273
		return KErrNotFound;
sl@0
   274
	// is the matching recognizer locked
sl@0
   275
	CApaDataRecognizerType* rec=iDataRecognizerList[i];
sl@0
   276
	if (rec->Locked())
sl@0
   277
		return KErrLocked;
sl@0
   278
	// remove the recognizer from the list, then delete it
sl@0
   279
	delete rec;
sl@0
   280
	iDataRecognizerList.Remove(i);
sl@0
   281
	iDataRecognizerList.Compress();
sl@0
   282
	return KErrNone;
sl@0
   283
	}
sl@0
   284
sl@0
   285
EXPORT_C void CApaDataRecognizer::DataTypeL(CDataTypeArray& aArray)
sl@0
   286
	{
sl@0
   287
	TInt count=iDataArray.Count();
sl@0
   288
	for (TInt i=0; i<count; i++)
sl@0
   289
		aArray.AppendL(iDataArray[i]);
sl@0
   290
	}
sl@0
   291
sl@0
   292
EXPORT_C void CApaDataRecognizer::UpdateDataTypesL()
sl@0
   293
// rebuild the list of unique mime types
sl@0
   294
	{
sl@0
   295
	iDataArray.Reset();
sl@0
   296
	TInt count=iDataRecognizerList.Count();
sl@0
   297
	for (TInt i=0; i<count; i++)
sl@0
   298
		{
sl@0
   299
		CApaDataRecognizerType* rec=iDataRecognizerList[i];
sl@0
   300
		TRAP_IGNORE(rec->UpdateDataTypesL());
sl@0
   301
		TInt mimeCount=rec->MimeTypesCount();
sl@0
   302
		for (TInt ii=0; ii<mimeCount; ii++)
sl@0
   303
			{
sl@0
   304
			TRAP_IGNORE(AddDataTypeL(rec->SupportedDataTypeL(ii)));
sl@0
   305
			}
sl@0
   306
		}
sl@0
   307
	}
sl@0
   308
sl@0
   309
EXPORT_C TInt CApaDataRecognizer::PreferredBufSize() const
sl@0
   310
// return the maximum preferred buf size
sl@0
   311
	{
sl@0
   312
	if (iMaxBufferSize < 0)
sl@0
   313
		{
sl@0
   314
		// The recognizers have been (re)loaded, so calulate the maximum buffer size.
sl@0
   315
		TInt count=iDataRecognizerList.Count();
sl@0
   316
		for (TInt i=0; i<count; i++)
sl@0
   317
			iMaxBufferSize=Max((TInt)iDataRecognizerList[i]->PreferredBufSize(), iMaxBufferSize);
sl@0
   318
		}
sl@0
   319
	return iMaxBufferSize;
sl@0
   320
	}
sl@0
   321
sl@0
   322
EXPORT_C void CApaDataRecognizer::DestroyRecognizerList()
sl@0
   323
	{
sl@0
   324
	iDataRecognizerList.ResetAndDestroy();
sl@0
   325
	iDataArray.Reset();
sl@0
   326
	}
sl@0
   327
sl@0
   328
void CApaDataRecognizer::AddDataTypeL(const TDataType& aDataType)
sl@0
   329
// add a mime type to the array if it's unique
sl@0
   330
	{
sl@0
   331
	TInt i=iDataArray.Count()-1;
sl@0
   332
	for (; i>=0; i--)
sl@0
   333
		{
sl@0
   334
		if (aDataType==iDataArray[i])
sl@0
   335
			return;
sl@0
   336
		}
sl@0
   337
	iDataArray.AppendL(aDataType);
sl@0
   338
	}
sl@0
   339
sl@0
   340
//
sl@0
   341
// class CApaDataRecognizerType
sl@0
   342
//
sl@0
   343
sl@0
   344
EXPORT_C CApaDataRecognizerType::CApaDataRecognizerType(TUid aUid, TInt aPriority)
sl@0
   345
	:iTypeUid(aUid),
sl@0
   346
	iPriority(aPriority),iDataRecognizerExtn(NULL)
sl@0
   347
/** Constructs the recognizer with a UID and a priority.
sl@0
   348
sl@0
   349
Typically, a derived class constructor calls this constructor through a constructor 
sl@0
   350
initialization list.
sl@0
   351
sl@0
   352
The UID is the way that a recognizer is detected by the framework.
sl@0
   353
sl@0
   354
@param aUid A UID that identifies the recognizer.
sl@0
   355
@param aPriority A value that estimates the probability that the recognizer 
sl@0
   356
will successfully identify data. This is one of the CApaDataRecognizerType::TRecognizerPriority 
sl@0
   357
enumerators.
sl@0
   358
@see CApaDataRecognizerType::TRecognizerPriority */
sl@0
   359
	{}
sl@0
   360
sl@0
   361
EXPORT_C CApaDataRecognizerType::~CApaDataRecognizerType()
sl@0
   362
// Destructor.
sl@0
   363
	{
sl@0
   364
#ifdef USING_ECOM_RECOGS
sl@0
   365
	//if ecom plugin is used destroy its implementation
sl@0
   366
	if(iDataRecognizerExtn!=NULL)
sl@0
   367
		{
sl@0
   368
		REComSession::DestroyedImplementation(iDataRecognizerExtn->DestructorUid());
sl@0
   369
		delete iDataRecognizerExtn;
sl@0
   370
		}
sl@0
   371
#endif
sl@0
   372
	}
sl@0
   373
sl@0
   374
EXPORT_C void CApaDataRecognizerType::Lock()
sl@0
   375
/** Adds a lock to the recognizer.
sl@0
   376
sl@0
   377
This may be called any number of times, but each call to this function must 
sl@0
   378
be matched by a corresponding call to Unlock() to completely unlock the recognizer.
sl@0
   379
sl@0
   380
This function is used to prevent the recognizer DLL from being unloaded.
sl@0
   381
sl@0
   382
@see Unlock()
sl@0
   383
@see Locked() */
sl@0
   384
	{
sl@0
   385
	iLock++;
sl@0
   386
	}
sl@0
   387
sl@0
   388
EXPORT_C void CApaDataRecognizerType::Unlock()
sl@0
   389
/** Removes a lock from the recognizer.
sl@0
   390
sl@0
   391
All calls to Lock() should be matched by a corresponding call to this function. 
sl@0
   392
The recognizer is not unlocked until all calls to Lock() have been matched 
sl@0
   393
by corresponding calls to this function.
sl@0
   394
sl@0
   395
@see Lock()
sl@0
   396
@see Locked() */
sl@0
   397
	{
sl@0
   398
	if (iLock>0)
sl@0
   399
		iLock--;
sl@0
   400
	}
sl@0
   401
sl@0
   402
EXPORT_C TUint CApaDataRecognizerType::PreferredBufSize()
sl@0
   403
/** Gets the size of buffer preferred for the purpose of recognizing the data type.
sl@0
   404
sl@0
   405
Regardless of the preferred buffer size returned by an implementation of this 
sl@0
   406
function, the actual size used is never greater than a maximum value as set 
sl@0
   407
by the client of the Application Architecture server through a call to RApaLsSession::SetMaxDataBufSize().
sl@0
   408
sl@0
   409
@return The preferred data size. The default implementation returns zero.
sl@0
   410
@see RApaLsSession::SetMaxDataBufSize() */
sl@0
   411
	{
sl@0
   412
	return 0;	// default to no buffer
sl@0
   413
				// name recognition only
sl@0
   414
	}
sl@0
   415
sl@0
   416
EXPORT_C void CApaDataRecognizerType::UpdateDataTypesL()
sl@0
   417
/** Refreshes the list of data (MIME) types supported by this recognizer. */
sl@0
   418
	{
sl@0
   419
	}
sl@0
   420
sl@0
   421
EXPORT_C void CApaDataRecognizerType::DoRecognizeL(const TDesC& aName, const TDesC8& aBuffer)
sl@0
   422
/** Implements the attempt to recognize data.
sl@0
   423
sl@0
   424
Recognizers should provide an implementation of this function in a derived 
sl@0
   425
class. Note that, when the implementation recognizes data, it must put the 
sl@0
   426
result of the operation in the iDataType and iConfidence data members.
sl@0
   427
sl@0
   428
Note that if more data is required than is provided in aBuffer, then 
sl@0
   429
CApaDataRecognizerType::FilePassedByHandleL should be called and the data 
sl@0
   430
read from the returned RFile (if not NULL). If this returns NULL, it may be 
sl@0
   431
possible to retrieve the data by calling RFile::Open() on aName, but only if 
sl@0
   432
aName is a legal file-name. It may be something else, such as a URL.
sl@0
   433
sl@0
   434
The default implementation does not recognize data.
sl@0
   435
sl@0
   436
@param aName The name of the data; typically this is a file name containing 
sl@0
   437
the data to be recognized. It is not necessarily a legal file-name though. It 
sl@0
   438
may, for example, be a URL/URI.
sl@0
   439
@param aBuffer A buffer containing data to be recognized; typically, this is 
sl@0
   440
read from the start of the file containing the data.
sl@0
   441
@see RecognizeL()
sl@0
   442
@see FilePassedByHandleL()
sl@0
   443
*/
sl@0
   444
	{
sl@0
   445
	(void)aName;
sl@0
   446
	(void)aBuffer;
sl@0
   447
	iDataType=TDataType();
sl@0
   448
	}
sl@0
   449
sl@0
   450
EXPORT_C RFile* CApaDataRecognizerType::FilePassedByHandleL()
sl@0
   451
/** Returns the RFile (if any) of file to be recognized.
sl@0
   452
sl@0
   453
This function returns the file passed by handle from the client-side (i.e. from calls to the RFile-parameter overloads of RApaLsSession's RecognizeData, RecognizeSpecificData, AppForDocument and StartDocument). The function returns NULL if the file to be recognized was not passed by handle.
sl@0
   454
sl@0
   455
It may only be called from implementations of DoRecognizeL - indeed the purpose of this function is logically to provide an extra parameter to the virtual DoRecognizeL function. All references/pointers to the 
sl@0
   456
RFile object returned must be discarded when the implementation of DoRecognizeL returns.
sl@0
   457
sl@0
   458
The RFile returned (if any) may be used by implementations of DoRecognizeL to retrieve more data than is provided in DoRecognizeL's aBuffer parameter.
sl@0
   459
sl@0
   460
The current-position of the returned RFile is the start of the file.
sl@0
   461
sl@0
   462
@return The file, passed by handle, to be recognized. Returns NULL if the data to be recognized was passed from the client-side by name/buffer rather than by file-handle. Ownership of the returned object is NOT transferred to the caller.
sl@0
   463
@see DoRecognizeL()
sl@0
   464
*/
sl@0
   465
	{
sl@0
   466
	RFile* const filePassedByHandle=iDataRecognizerExtn->DataRecognizer().FilePassedByHandle();
sl@0
   467
	if (filePassedByHandle!=NULL && (filePassedByHandle->SubSessionHandle()!=KNullHandle))
sl@0
   468
		{
sl@0
   469
		// set the current file-position to the start of the file
sl@0
   470
		TInt filePosition=0;
sl@0
   471
		User::LeaveIfError(filePassedByHandle->Seek(ESeekStart, filePosition));
sl@0
   472
		}
sl@0
   473
	return filePassedByHandle;
sl@0
   474
	}
sl@0
   475
sl@0
   476
EXPORT_C TDataRecognitionResult CApaDataRecognizerType::RecognizeL(const TDesC& aName, const TDesC8& aBuffer)
sl@0
   477
/** Attempts to recognize data.
sl@0
   478
sl@0
   479
This function is called by the Application Architecture server as a result 
sl@0
   480
of client calls to the server through an instance of RApaLsSession.
sl@0
   481
sl@0
   482
The function calls DoRecognizeL() which implements recognition behaviour.
sl@0
   483
sl@0
   484
@param aName The name of the data; typically this is a file name containing 
sl@0
   485
the data to be recognized.
sl@0
   486
@param aBuffer A buffer containing data to be recognized; typically, this is 
sl@0
   487
read from the start of the file containing the data. Implement PreferredBufSize() 
sl@0
   488
to define the ideal size for this buffer. Note that failure to implement PreferredBufSize() 
sl@0
   489
results in a default buffer size of zero.
sl@0
   490
@return The result of the attempt to recognize the data.
sl@0
   491
@see PreferredBufSize()
sl@0
   492
@see RApaLsSession
sl@0
   493
@see RApaLsSession::SetMaxDataBufSize() */
sl@0
   494
	{
sl@0
   495
	TDataToRecognize dataToRecognize(aName, aBuffer);
sl@0
   496
	return RecognizeL(dataToRecognize);
sl@0
   497
	}
sl@0
   498
sl@0
   499
TDataRecognitionResult CApaDataRecognizerType::RecognizeL(TDataToRecognize& aDataToRecognize)
sl@0
   500
/**
sl@0
   501
@internalComponent
sl@0
   502
*/
sl@0
   503
	{
sl@0
   504
	iDataType=TDataType();
sl@0
   505
	iConfidence=ENotRecognized;
sl@0
   506
	aDataToRecognize.RecognizeL(*this);
sl@0
   507
	TDataRecognitionResult rec;
sl@0
   508
	rec.iDataType=iDataType;
sl@0
   509
	rec.iConfidence=iConfidence;
sl@0
   510
	return rec;
sl@0
   511
	}
sl@0
   512
sl@0
   513
EXPORT_C TDataType CApaDataRecognizerType::MimeType()
sl@0
   514
/** Gets the data (MIME) type of the most recently recognized data.
sl@0
   515
sl@0
   516
@return The data (MIME) type.
sl@0
   517
@see iDataType */
sl@0
   518
	{
sl@0
   519
	return iDataType;
sl@0
   520
	}
sl@0
   521
sl@0
   522
EXPORT_C void CApaDataRecognizerType::Reserved_1()
sl@0
   523
// a spare empty virtual function
sl@0
   524
	{}
sl@0
   525
sl@0
   526
// instantiate the ecom implementation class 
sl@0
   527
EXPORT_C CApaDataRecognizerType* CApaDataRecognizerType::CreateDataRecognizerL(TUid aImplUid)
sl@0
   528
	{
sl@0
   529
#if !defined(USING_ECOM_RECOGS)
sl@0
   530
	(void)aImplUid;
sl@0
   531
	return NULL;
sl@0
   532
#endif
sl@0
   533
	CDataRecognizerExtension* const dataRecognizerExtn=CDataRecognizerExtension::NewL();
sl@0
   534
	CleanupStack::PushL(dataRecognizerExtn);
sl@0
   535
sl@0
   536
	CApaDataRecognizerType* const dataRecType=static_cast <CApaDataRecognizerType*>(REComSession::CreateImplementationL(aImplUid, dataRecognizerExtn->DestructorUidReference()));
sl@0
   537
sl@0
   538
	dataRecType->iDataRecognizerExtn = dataRecognizerExtn;
sl@0
   539
	CleanupStack::Pop(dataRecognizerExtn);
sl@0
   540
sl@0
   541
	return dataRecType;
sl@0
   542
	}
sl@0
   543
sl@0
   544
CDataRecognizerExtension* CApaDataRecognizerType::DataRecognizerExtension()
sl@0
   545
/** @internalComponent */
sl@0
   546
	{
sl@0
   547
	return iDataRecognizerExtn;
sl@0
   548
	}
sl@0
   549
sl@0
   550
//
sl@0
   551
// class TDataRecognitionResult
sl@0
   552
//
sl@0
   553
sl@0
   554
EXPORT_C void TDataRecognitionResult::Reset()
sl@0
   555
/** Resets the data type to the default data type and sets the confidence rating 
sl@0
   556
to CApaDataRecognizerType::ENotRecognized.
sl@0
   557
sl@0
   558
@see CApaDataRecognizerType::TRecognitionConfidence
sl@0
   559
@see TDataType */
sl@0
   560
	{
sl@0
   561
	iDataType=TDataType();
sl@0
   562
	iConfidence=CApaDataRecognizerType::ENotRecognized;
sl@0
   563
	}
sl@0
   564
sl@0
   565
//
sl@0
   566
// class TDataToRecognize
sl@0
   567
//
sl@0
   568
sl@0
   569
TDataToRecognize::TDataToRecognize(const TDesC& aName, const TDesC8& aBuffer)
sl@0
   570
	:iFlags(0),
sl@0
   571
	 iName(&aName),
sl@0
   572
	 iBuffer(&aBuffer),
sl@0
   573
	 iDataRecognizer(NULL),
sl@0
   574
	 iFile(NULL),
sl@0
   575
	 iFilePosition(0),
sl@0
   576
	 iPreferredBufSize(0)
sl@0
   577
	{
sl@0
   578
	__ASSERT_DEBUG(aName.Length()>0 || aBuffer.Length()>0, Panic(EDPanicInvalidData));
sl@0
   579
	}
sl@0
   580
sl@0
   581
TDataToRecognize::TDataToRecognize(CApaDataRecognizer& aDataRecognizer, RFile& aFile, TInt aPreferredBufSize)
sl@0
   582
	:iFlags(0),
sl@0
   583
	 iName(NULL),
sl@0
   584
	 iBuffer(NULL),
sl@0
   585
	 iDataRecognizer(&aDataRecognizer),
sl@0
   586
	 iFile(&aFile),
sl@0
   587
	 iFilePosition(0),
sl@0
   588
	 iPreferredBufSize(aPreferredBufSize)
sl@0
   589
	{
sl@0
   590
	}
sl@0
   591
sl@0
   592
void TDataToRecognize::RecognizeL(CApaDataRecognizerType& aDataRecognizerType)
sl@0
   593
	{
sl@0
   594
	__ASSERT_DEBUG(iName!=NULL, Panic(EDPanicNullPointer1));
sl@0
   595
	__ASSERT_DEBUG(iBuffer!=NULL, Panic(EDPanicNullPointer2));
sl@0
   596
	aDataRecognizerType.DoRecognizeL(*iName, *iBuffer);
sl@0
   597
	}
sl@0
   598
sl@0
   599
void TDataToRecognize::PrepareForRecognizeLoopLC()
sl@0
   600
	{
sl@0
   601
	CleanupStack::PushL(TCleanupItem(ResetAfterRecognizeLoopL, this));
sl@0
   602
	if (iFile!=NULL)
sl@0
   603
		{
sl@0
   604
		// set the file-pointer returned by CApaDataRecognizerType::FilePassedByHandleL
sl@0
   605
		iDataRecognizer->SetFilePassedByHandle(iFile);
sl@0
   606
sl@0
   607
		// store current-position in iFilePosition
sl@0
   608
		__ASSERT_DEBUG((iFlags&EFlagFilePositionStored)==0, Panic(EDPanicBadFlagState1));
sl@0
   609
		iFlags|=EFlagFilePositionStored;
sl@0
   610
		iFilePosition=0;
sl@0
   611
		User::LeaveIfError(iFile->Seek(ESeekCurrent, iFilePosition));
sl@0
   612
sl@0
   613
		// read a buffer of the appropriate length from the start of the file
sl@0
   614
		__ASSERT_DEBUG(iBuffer==NULL, Panic(EDPanicNullPointerExpected1));
sl@0
   615
		TInt filePosition=0;
sl@0
   616
		User::LeaveIfError(iFile->Seek(ESeekStart, filePosition));
sl@0
   617
		HBufC8* const buffer=HBufC8::NewL(iPreferredBufSize);
sl@0
   618
		iBuffer=buffer; // iBuffer now "owns" this (as long as iFile!=NULL, which is true)
sl@0
   619
		TPtr8 buffer_asWritable(buffer->Des());
sl@0
   620
		User::LeaveIfError(iFile->Read(buffer_asWritable));
sl@0
   621
sl@0
   622
		// prepend "::" to the file-name to be passed to DoRecognizeL to make it an illegal file-name - this
sl@0
   623
		// is to prevent any behavioural break since implementers of DoRecognizeL may assume that if aName is
sl@0
   624
		// *not* an illegal file-name they can pass it to RFile::Open, which will not be possible if we're
sl@0
   625
		// recognizing by file-handle, both because of data-caging (the file may not live in a directory that
sl@0
   626
		// the Apparc process has permission to open it from) and because aName does not contain the full path
sl@0
   627
		__ASSERT_DEBUG(iName==NULL, Panic(EDPanicNullPointerExpected2));
sl@0
   628
		TDes* const name=new(ELeave) TFileName;
sl@0
   629
		iName=name; // iName now "owns" this (as long as iFile!=NULL, which is true)
sl@0
   630
		User::LeaveIfError(iFile->Name(*name));
sl@0
   631
		_LIT(KLitIllegalifier, "::");
sl@0
   632
		name->Insert(0, KLitIllegalifier);
sl@0
   633
		}
sl@0
   634
	}
sl@0
   635
sl@0
   636
void TDataToRecognize::PopOffCleanupStackAndResetAfterRecognizeLoopL()
sl@0
   637
	{
sl@0
   638
	iFlags|=EFlagLeaveIfError; // we don't want to leave from ResetAfterRecognizeLoopL if we're being cleaned up from the clean-up stack because of a leave, but if we're leaving because of a CleanupStack::PopAndDestroy then it's okay to leave
sl@0
   639
	CleanupStack::PopAndDestroy(this); // calls ResetAfterRecognizeLoopL
sl@0
   640
	}
sl@0
   641
sl@0
   642
void TDataToRecognize::ResetAfterRecognizeLoopL(TAny* aThis)
sl@0
   643
	{ // static
sl@0
   644
	STATIC_CAST(TDataToRecognize*, aThis)->ResetAfterRecognizeLoopL();
sl@0
   645
	}
sl@0
   646
sl@0
   647
void TDataToRecognize::ResetAfterRecognizeLoopL()
sl@0
   648
	{
sl@0
   649
	TInt error=KErrNone;
sl@0
   650
	if (iFile!=NULL)
sl@0
   651
		{
sl@0
   652
		// NULLify the file-pointer returned by CApaDataRecognizerType::FilePassedByHandleL
sl@0
   653
		iDataRecognizer->SetFilePassedByHandle(NULL);
sl@0
   654
sl@0
   655
		// reset the current-position of the file
sl@0
   656
		__ASSERT_DEBUG(iFlags&EFlagFilePositionStored, Panic(EDPanicBadFlagState2));
sl@0
   657
		error=iFile->Seek(ESeekStart, iFilePosition);
sl@0
   658
		iFlags&=~EFlagFilePositionStored;
sl@0
   659
sl@0
   660
		// delete the objects that are owned if iFile!=NULL (which is true)
sl@0
   661
		delete iName;
sl@0
   662
		delete iBuffer;
sl@0
   663
		}
sl@0
   664
	if (iFlags&EFlagLeaveIfError)
sl@0
   665
		{
sl@0
   666
		iFlags&=~EFlagLeaveIfError; // probably not necessary, just defensive programming
sl@0
   667
		User::LeaveIfError(error);
sl@0
   668
		}
sl@0
   669
	}
sl@0
   670