os/ossrv/lowlevellibsandfws/apputils/src/BACLIPB.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
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
// Started by DS, October 1996
sl@0
    15
// Clipboard
sl@0
    16
// 
sl@0
    17
//
sl@0
    18
sl@0
    19
#include <baclipb.h>
sl@0
    20
#include <s32file.h>
sl@0
    21
#include <basched.h>
sl@0
    22
#include <bautils.h>
sl@0
    23
#include <s32stor.h>
sl@0
    24
#include <hal.h>
sl@0
    25
#include <hal_data.h>
sl@0
    26
#include <baflpan.h>
sl@0
    27
sl@0
    28
sl@0
    29
#ifdef _UNICODE
sl@0
    30
const TUid KClipboardFileUid16={0x10003A10};
sl@0
    31
#define KClipboardFileUid KClipboardFileUid16
sl@0
    32
#else
sl@0
    33
const TUid KClipboardFileUid8={268435515};
sl@0
    34
#define KClipboardFileUid KClipboardFileUid8
sl@0
    35
#endif
sl@0
    36
sl@0
    37
_LIT(KClipboardFileName,"\\System\\Data\\Clpboard.cbd");
sl@0
    38
sl@0
    39
sl@0
    40
inline CClipboard::CClipboard(RFs& aFs)
sl@0
    41
	: iFs(aFs)
sl@0
    42
	{__DECLARE_NAME(_S("CClipboard"));}
sl@0
    43
sl@0
    44
CClipboard* CClipboard::NewLC(RFs& aFs)
sl@0
    45
	{
sl@0
    46
	CClipboard* self=new(ELeave) CClipboard(aFs);
sl@0
    47
    CleanupStack::PushL(self);
sl@0
    48
	return self;
sl@0
    49
	}
sl@0
    50
sl@0
    51
void CClipboard::ConstructReadL()
sl@0
    52
	{
sl@0
    53
	__ASSERT_DEBUG(iStore==NULL&&iStreamDictionary==NULL,User::Invariant());
sl@0
    54
sl@0
    55
//	
sl@0
    56
	TDriveName clipboardDrive = ClipboardFileDrive();
sl@0
    57
	TFileName clipboardFile(KClipboardFileName);
sl@0
    58
	clipboardFile.Insert(0,clipboardDrive);
sl@0
    59
sl@0
    60
	CFileStore* store=CDirectFileStore::OpenLC(iFs,clipboardFile,EFileRead|EFileShareReadersOnly);
sl@0
    61
	if (store->Type()!=TUidType(KDirectFileStoreLayoutUid,KClipboardFileUid))
sl@0
    62
		User::Leave(KErrCorrupt);
sl@0
    63
	CStreamDictionary* dict=CStreamDictionary::NewLC();
sl@0
    64
	RStoreReadStream root;
sl@0
    65
	root.OpenLC(*store,store->Root());
sl@0
    66
	root >> *dict;
sl@0
    67
	CleanupStack::PopAndDestroy(); // root stream
sl@0
    68
//
sl@0
    69
	CleanupStack::Pop(2);	// store and dictionary
sl@0
    70
	iStore=store;
sl@0
    71
	iStreamDictionary=dict;
sl@0
    72
	}
sl@0
    73
sl@0
    74
sl@0
    75
EXPORT_C TDriveName CClipboard::ClipboardFileDrive()
sl@0
    76
/**
sl@0
    77
 Returns the clipboad file drive information stored in the TLS
sl@0
    78
 
sl@0
    79
 @return A drive letter and a colon (A:, B:... etc,.)
sl@0
    80
 @panic BAFL 23 If drive is either not in range 0-24 or is not defined in HAL file(s).
sl@0
    81
 If this panic occurs, check the value of EClipboardDrive defined in HAL file(s)
sl@0
    82
 i,e <values.hda> & <config.hcf>. This value should be in range EDriveA-EDriveY.
sl@0
    83
 */
sl@0
    84
    {
sl@0
    85
	TInt drive;
sl@0
    86
	TInt retVal;
sl@0
    87
	TBool badDrive = ETrue;
sl@0
    88
sl@0
    89
	retVal = HAL::Get(HAL::EClipboardDrive,drive);	
sl@0
    90
	
sl@0
    91
	if(retVal == KErrNone)
sl@0
    92
		{
sl@0
    93
		//EDriveZ can not be a valid clipboard drive as it is non-writable
sl@0
    94
		// Check if drive is between range 0-24,
sl@0
    95
		if(EDriveA <= drive && EDriveY >= drive)
sl@0
    96
			{
sl@0
    97
			badDrive = EFalse; // Drive is valid				
sl@0
    98
			}
sl@0
    99
		}
sl@0
   100
			
sl@0
   101
	__ASSERT_ALWAYS((badDrive == EFalse), ::Panic(EBafPanicBadDrive));
sl@0
   102
sl@0
   103
	TDriveUnit driveUnit(drive);
sl@0
   104
	return driveUnit.Name();
sl@0
   105
	}
sl@0
   106
sl@0
   107
EXPORT_C CClipboard* CClipboard::NewForReadingLC(RFs& aFs)
sl@0
   108
/** Constructs a clipboard object and prepares the clipboard's store for reading, 
sl@0
   109
placing a pointer to the object on the cleanup stack. This allows the object 
sl@0
   110
and allocated resources to be cleaned up if a subsequent leave occurs.
sl@0
   111
sl@0
   112
@param aFs A handle to a file server session. 
sl@0
   113
@return A pointer to the newly constructed clipboard object. This pointer is 
sl@0
   114
put onto the cleanup stack. */
sl@0
   115
    {
sl@0
   116
	CClipboard* self=NewLC(aFs);
sl@0
   117
	TRAPD(error,self->ConstructReadL());
sl@0
   118
	switch (error)
sl@0
   119
		{
sl@0
   120
	default:
sl@0
   121
		User::LeaveIfError(error);
sl@0
   122
		__ASSERT_DEBUG(self->iStore!=NULL&&self->iStreamDictionary!=NULL,User::Invariant());
sl@0
   123
		break;
sl@0
   124
	case KErrPathNotFound:	// clipboard does not exist
sl@0
   125
	case KErrNotFound:		// no root stream
sl@0
   126
	case KErrInUse:			// someone is writing to the clipboard
sl@0
   127
	case KErrCorrupt:		// type is invalid, or cardinality failure
sl@0
   128
	case KErrEof:			// wrong structure
sl@0
   129
	case KErrNotSupported:	// not a file store
sl@0
   130
		// return an empty clipboard: will need an empty dictionary
sl@0
   131
		__ASSERT_DEBUG(self->iStore==NULL&&self->iStreamDictionary==NULL,User::Invariant());
sl@0
   132
		self->iStreamDictionary=CStreamDictionary::NewL();
sl@0
   133
		break;
sl@0
   134
		}
sl@0
   135
	return(self);
sl@0
   136
	}
sl@0
   137
sl@0
   138
EXPORT_C CClipboard* CClipboard::NewForReadingL(RFs& aFs)
sl@0
   139
/** Constructs a clipboard object and prepares the clipboard's store for reading.
sl@0
   140
sl@0
   141
@param aFs A handle to a file server session. 
sl@0
   142
@return A pointer to the newly constructed clipboard object. */
sl@0
   143
	{
sl@0
   144
	CClipboard* self=CClipboard::NewForReadingLC(aFs);
sl@0
   145
	CleanupStack::Pop();
sl@0
   146
	return self;
sl@0
   147
	}
sl@0
   148
sl@0
   149
EXPORT_C CClipboard* CClipboard::NewForWritingLC(RFs& aFs)
sl@0
   150
/** Constructs a clipboard object and prepares the clipboard's store for writing.
sl@0
   151
sl@0
   152
If the file associated with the clipboard's store does not exist, it is created; 
sl@0
   153
if it already exists, any existing content is discarded.
sl@0
   154
sl@0
   155
@param aFs A handle to a file server session. 
sl@0
   156
@return A pointer to the newly constructed clipboard object. This pointer is 
sl@0
   157
put onto the cleanup stack. */
sl@0
   158
    {
sl@0
   159
	CClipboard* self=NewLC(aFs);
sl@0
   160
	self->iStreamDictionary=CStreamDictionary::NewL();
sl@0
   161
//
sl@0
   162
	TDriveName clipboardDrive = ClipboardFileDrive();
sl@0
   163
	TFileName clipboardFile(KClipboardFileName);
sl@0
   164
	clipboardFile.Insert(0,clipboardDrive);
sl@0
   165
sl@0
   166
	RFile file;
sl@0
   167
	TInt r=file.Open(aFs,clipboardFile,EFileRead|EFileWrite);
sl@0
   168
sl@0
   169
	/**
sl@0
   170
	 * The following branching is desired at this stage:
sl@0
   171
	 * -> if the path is not present, we need to create it and then create the file. 
sl@0
   172
	 * -> if the path is present, we only need to create the file 
sl@0
   173
	 * -> if both path and file exists, we don't need to do anything 
sl@0
   174
	 * 
sl@0
   175
	 * Instead of the switch case block we can achieve this by atleast 2 other means.
sl@0
   176
 	 * Option 1:
sl@0
   177
	 * RFile file; 
sl@0
   178
	 * TInt r=file.Open(aFs,clipboardFile,EFileRead|EFileWrite); 
sl@0
   179
	 * if(r == KErrPathNotFound) { 
sl@0
   180
	 *       BaflUtils::EnsurePathExistsL(aFs,clipboardFile); 
sl@0
   181
	 *       r=file.Replace(aFs,clipboardFile,EFileRead|EFileWrite); 
sl@0
   182
	 * } else if(r == KErrNotFound) { 
sl@0
   183
	 *       r=file.Replace(aFs,clipboardFile,EFileRead|EFileWrite); 
sl@0
   184
	 * }
sl@0
   185
	 * User::LeaveIfError(r);
sl@0
   186
	 *
sl@0
   187
	 * => the above code has extra check to be made when the error is KErrNotFound 
sl@0
   188
	 *    and has increased code size owing to repeated file.Replace().
sl@0
   189
	 *
sl@0
   190
	 * Option 2:
sl@0
   191
	 * BaflUtils::EnsurePathExistsL(aFs,clipboardFile); 
sl@0
   192
	 * RFile file; 
sl@0
   193
	 * TInt r=file.Open(aFs,clipboardFile,EFileRead|EFileWrite); 
sl@0
   194
	 * if(r == KErrNotFound) { 
sl@0
   195
	 *        r=file.Replace(aFs,clipboardFile,EFileRead|EFileWrite); 
sl@0
   196
	 * } 
sl@0
   197
	 * User::LeaveIfError(r);
sl@0
   198
	 * => Call to BaflUtils::EnsurePathExistsL unconditionally which in turn 
sl@0
   199
	 * invokes Rfs::MkDirAll() can be expensive and errorneous.
sl@0
   200
	 * (Check documnentation for RFs::MkDirAll()
sl@0
   201
	 */
sl@0
   202
 
sl@0
   203
sl@0
   204
	//Maintain the order of the switch case and DO NOT add break between the
sl@0
   205
	//switch cases. They have been written such that only the required set of 
sl@0
   206
	//functionality is executed based on the Error code.
sl@0
   207
	switch (r)
sl@0
   208
		{
sl@0
   209
	case KErrPathNotFound:		// path has not been created yet
sl@0
   210
		BaflUtils::EnsurePathExistsL(aFs,clipboardFile);
sl@0
   211
		// coverity [fallthrough]
sl@0
   212
		// Missing break intentional. Read comment before switch case.
sl@0
   213
	case KErrNotFound:			// it does not exist yet
sl@0
   214
		r=file.Replace(aFs,clipboardFile,EFileRead|EFileWrite);
sl@0
   215
		// coverity [fallthrough]
sl@0
   216
		// Missing break intentional. Read comment before switch case.
sl@0
   217
	default:
sl@0
   218
		User::LeaveIfError(r);
sl@0
   219
		}
sl@0
   220
//
sl@0
   221
	CFileStore* store=CDirectFileStore::NewL(file);
sl@0
   222
	self->iStore=store;
sl@0
   223
	store->SetTypeL(TUidType(KDirectFileStoreLayoutUid,KClipboardFileUid));
sl@0
   224
	return self;
sl@0
   225
	}
sl@0
   226
sl@0
   227
EXPORT_C TInt CClipboard::Clear(RFs& aFs)
sl@0
   228
/** Empties the clipboard.
sl@0
   229
sl@0
   230
Note that if the file associated with the clipboard's store does not exist, 
sl@0
   231
this is not regarded as an error and the function completes successfully.
sl@0
   232
sl@0
   233
@param aFs A handle to a file server session. 
sl@0
   234
@return KErrNone if successful, otherwise another of the system-wide error 
sl@0
   235
codes. */
sl@0
   236
	{
sl@0
   237
	TDriveName clipboardDrive = ClipboardFileDrive();
sl@0
   238
	TFileName clipboardFile(KClipboardFileName);
sl@0
   239
	clipboardFile.Insert(0,clipboardDrive);
sl@0
   240
sl@0
   241
	RFile file;
sl@0
   242
	TInt r=file.Open(aFs,clipboardFile,EFileWrite);
sl@0
   243
	switch (r)
sl@0
   244
		{
sl@0
   245
	case KErrNone:			// clear without notification
sl@0
   246
		r=file.SetSize(0);
sl@0
   247
		file.Close();
sl@0
   248
		break;
sl@0
   249
	case KErrPathNotFound:	// no clipboard to clear
sl@0
   250
	case KErrNotFound:
sl@0
   251
		r=KErrNone;
sl@0
   252
		break;
sl@0
   253
	default:
sl@0
   254
		r=aFs.Delete(clipboardFile);	// just delete the fella
sl@0
   255
		break;
sl@0
   256
		}
sl@0
   257
	return r;
sl@0
   258
	}
sl@0
   259
sl@0
   260
EXPORT_C CClipboard::~CClipboard()
sl@0
   261
/** Destructor. Frees all resources owned by the object, prior to its destruction. 
sl@0
   262
In particular, the file associated with the clipboard's store is closed.
sl@0
   263
sl@0
   264
Note that if the clipboard file store has no root stream, i.e. the CommitL() 
sl@0
   265
member function of CClipboard has not been called prior to deleting the CClipboard 
sl@0
   266
object, then the file associated with the clipboard file store is deleted. */
sl@0
   267
    {
sl@0
   268
	delete iStreamDictionary;
sl@0
   269
	CFileStore* store=iStore;
sl@0
   270
	if (store!=NULL)
sl@0
   271
		{
sl@0
   272
		TBool rollback=(store->Root()==KNullStreamId);
sl@0
   273
   		delete store;
sl@0
   274
		if (rollback)	// useless clipboard, no root stream
sl@0
   275
			{
sl@0
   276
			TDriveName clipboardDrive = ClipboardFileDrive();
sl@0
   277
			TFileName clipboardFile(KClipboardFileName);
sl@0
   278
			clipboardFile.Insert(0,clipboardDrive);
sl@0
   279
			
sl@0
   280
			iFs.Delete(clipboardFile);
sl@0
   281
			}
sl@0
   282
		}
sl@0
   283
    }
sl@0
   284
sl@0
   285
EXPORT_C void CClipboard::CommitL()
sl@0
   286
/** Commits changes to the clipboard's store.
sl@0
   287
sl@0
   288
It externalises the stream dictionary to the clipboard store as the root stream 
sl@0
   289
and then commits all changes to the store. This function must be called after 
sl@0
   290
application data has been externalised; failure to do so results in the deletion 
sl@0
   291
of the file associated with the clipboard store and the loss of data. */
sl@0
   292
	{
sl@0
   293
	__ASSERT_DEBUG(iStore!=NULL&&iStreamDictionary!=NULL,User::Invariant());
sl@0
   294
//
sl@0
   295
	CFileStore& store=*iStore;
sl@0
   296
	RStoreWriteStream stream;
sl@0
   297
	TStreamId root=stream.CreateLC(store);
sl@0
   298
	stream<< *iStreamDictionary;
sl@0
   299
	stream.CommitL();
sl@0
   300
	CleanupStack::PopAndDestroy(); // dictionary stream
sl@0
   301
	store.SetRootL(root);
sl@0
   302
	store.CommitL();
sl@0
   303
	}
sl@0
   304
sl@0
   305
sl@0
   306
// Copy and paste member functions for low level data types
sl@0
   307
sl@0
   308
sl@0
   309
const TUid KClipboardRealTypeUid = { 268435705 };
sl@0
   310
sl@0
   311
sl@0
   312
EXPORT_C void  CClipboard::CopyToL(TReal aReal) __SOFTFP
sl@0
   313
/** Copies a double-precision floating point value to the clipboard.
sl@0
   314
sl@0
   315
Note that the function does not automatically commit changes to the clipboard's 
sl@0
   316
store. This must be done explicitly.
sl@0
   317
sl@0
   318
@param aReal The double-precision floating point value to be copied to the 
sl@0
   319
clipboard.
sl@0
   320
@see CommitL() */
sl@0
   321
    {
sl@0
   322
	__ASSERT_DEBUG(iStore!=NULL&&iStreamDictionary!=NULL,User::Invariant());
sl@0
   323
//
sl@0
   324
    RStoreWriteStream stream;
sl@0
   325
    TStreamId id = stream.CreateLC ( Store() );
sl@0
   326
    stream << aReal;
sl@0
   327
	stream.CommitL();
sl@0
   328
    CleanupStack::PopAndDestroy();
sl@0
   329
    StreamDictionary().AssignL ( KClipboardRealTypeUid, id );
sl@0
   330
    }
sl@0
   331
sl@0
   332
EXPORT_C TBool CClipboard::PasteFromL  ( TReal& aReal )
sl@0
   333
/** Pastes a double-precision floating point value from the clipboard.
sl@0
   334
sl@0
   335
If a double-precision floating point value exists on the clipboard, then the 
sl@0
   336
function restores it to the referenced argument and returns a true value.
sl@0
   337
sl@0
   338
If there is no double-precision floating point value on the clipboard, then 
sl@0
   339
the function returns a false value. The referenced argument is not changed.
sl@0
   340
sl@0
   341
@param aReal On return, contains the double-precision floating point value 
sl@0
   342
found on the clipboard.
sl@0
   343
@return ETrue, if a double-precision floating point value exists on the 
sl@0
   344
clipboard and has been pasted to the referenced argument; EFalse otherwise. */
sl@0
   345
    {
sl@0
   346
	__ASSERT_DEBUG(iStreamDictionary!=NULL,User::Invariant());
sl@0
   347
    TStreamId id = StreamDictionary().At(KClipboardRealTypeUid);
sl@0
   348
    if (id == KNullStreamId)
sl@0
   349
		return (EFalse); // nothing here
sl@0
   350
    __ASSERT_DEBUG(iStore!=NULL,User::Invariant());
sl@0
   351
	RStoreReadStream stream;
sl@0
   352
    stream.OpenLC (Store(), id);
sl@0
   353
    stream >> aReal;
sl@0
   354
    CleanupStack::PopAndDestroy();
sl@0
   355
    return (ETrue);
sl@0
   356
    }
sl@0
   357
sl@0
   358
sl@0
   359
sl@0
   360
// end of BACLIPB.CPP