sl@0: // Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // Started by DS, October 1996 sl@0: // Clipboard sl@0: // sl@0: // sl@0: sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: sl@0: sl@0: #ifdef _UNICODE sl@0: const TUid KClipboardFileUid16={0x10003A10}; sl@0: #define KClipboardFileUid KClipboardFileUid16 sl@0: #else sl@0: const TUid KClipboardFileUid8={268435515}; sl@0: #define KClipboardFileUid KClipboardFileUid8 sl@0: #endif sl@0: sl@0: _LIT(KClipboardFileName,"\\System\\Data\\Clpboard.cbd"); sl@0: sl@0: sl@0: inline CClipboard::CClipboard(RFs& aFs) sl@0: : iFs(aFs) sl@0: {__DECLARE_NAME(_S("CClipboard"));} sl@0: sl@0: CClipboard* CClipboard::NewLC(RFs& aFs) sl@0: { sl@0: CClipboard* self=new(ELeave) CClipboard(aFs); sl@0: CleanupStack::PushL(self); sl@0: return self; sl@0: } sl@0: sl@0: void CClipboard::ConstructReadL() sl@0: { sl@0: __ASSERT_DEBUG(iStore==NULL&&iStreamDictionary==NULL,User::Invariant()); sl@0: sl@0: // sl@0: TDriveName clipboardDrive = ClipboardFileDrive(); sl@0: TFileName clipboardFile(KClipboardFileName); sl@0: clipboardFile.Insert(0,clipboardDrive); sl@0: sl@0: CFileStore* store=CDirectFileStore::OpenLC(iFs,clipboardFile,EFileRead|EFileShareReadersOnly); sl@0: if (store->Type()!=TUidType(KDirectFileStoreLayoutUid,KClipboardFileUid)) sl@0: User::Leave(KErrCorrupt); sl@0: CStreamDictionary* dict=CStreamDictionary::NewLC(); sl@0: RStoreReadStream root; sl@0: root.OpenLC(*store,store->Root()); sl@0: root >> *dict; sl@0: CleanupStack::PopAndDestroy(); // root stream sl@0: // sl@0: CleanupStack::Pop(2); // store and dictionary sl@0: iStore=store; sl@0: iStreamDictionary=dict; sl@0: } sl@0: sl@0: sl@0: EXPORT_C TDriveName CClipboard::ClipboardFileDrive() sl@0: /** sl@0: Returns the clipboad file drive information stored in the TLS sl@0: sl@0: @return A drive letter and a colon (A:, B:... etc,.) sl@0: @panic BAFL 23 If drive is either not in range 0-24 or is not defined in HAL file(s). sl@0: If this panic occurs, check the value of EClipboardDrive defined in HAL file(s) sl@0: i,e & . This value should be in range EDriveA-EDriveY. sl@0: */ sl@0: { sl@0: TInt drive; sl@0: TInt retVal; sl@0: TBool badDrive = ETrue; sl@0: sl@0: retVal = HAL::Get(HAL::EClipboardDrive,drive); sl@0: sl@0: if(retVal == KErrNone) sl@0: { sl@0: //EDriveZ can not be a valid clipboard drive as it is non-writable sl@0: // Check if drive is between range 0-24, sl@0: if(EDriveA <= drive && EDriveY >= drive) sl@0: { sl@0: badDrive = EFalse; // Drive is valid sl@0: } sl@0: } sl@0: sl@0: __ASSERT_ALWAYS((badDrive == EFalse), ::Panic(EBafPanicBadDrive)); sl@0: sl@0: TDriveUnit driveUnit(drive); sl@0: return driveUnit.Name(); sl@0: } sl@0: sl@0: EXPORT_C CClipboard* CClipboard::NewForReadingLC(RFs& aFs) sl@0: /** Constructs a clipboard object and prepares the clipboard's store for reading, sl@0: placing a pointer to the object on the cleanup stack. This allows the object sl@0: and allocated resources to be cleaned up if a subsequent leave occurs. sl@0: sl@0: @param aFs A handle to a file server session. sl@0: @return A pointer to the newly constructed clipboard object. This pointer is sl@0: put onto the cleanup stack. */ sl@0: { sl@0: CClipboard* self=NewLC(aFs); sl@0: TRAPD(error,self->ConstructReadL()); sl@0: switch (error) sl@0: { sl@0: default: sl@0: User::LeaveIfError(error); sl@0: __ASSERT_DEBUG(self->iStore!=NULL&&self->iStreamDictionary!=NULL,User::Invariant()); sl@0: break; sl@0: case KErrPathNotFound: // clipboard does not exist sl@0: case KErrNotFound: // no root stream sl@0: case KErrInUse: // someone is writing to the clipboard sl@0: case KErrCorrupt: // type is invalid, or cardinality failure sl@0: case KErrEof: // wrong structure sl@0: case KErrNotSupported: // not a file store sl@0: // return an empty clipboard: will need an empty dictionary sl@0: __ASSERT_DEBUG(self->iStore==NULL&&self->iStreamDictionary==NULL,User::Invariant()); sl@0: self->iStreamDictionary=CStreamDictionary::NewL(); sl@0: break; sl@0: } sl@0: return(self); sl@0: } sl@0: sl@0: EXPORT_C CClipboard* CClipboard::NewForReadingL(RFs& aFs) sl@0: /** Constructs a clipboard object and prepares the clipboard's store for reading. sl@0: sl@0: @param aFs A handle to a file server session. sl@0: @return A pointer to the newly constructed clipboard object. */ sl@0: { sl@0: CClipboard* self=CClipboard::NewForReadingLC(aFs); sl@0: CleanupStack::Pop(); sl@0: return self; sl@0: } sl@0: sl@0: EXPORT_C CClipboard* CClipboard::NewForWritingLC(RFs& aFs) sl@0: /** Constructs a clipboard object and prepares the clipboard's store for writing. sl@0: sl@0: If the file associated with the clipboard's store does not exist, it is created; sl@0: if it already exists, any existing content is discarded. sl@0: sl@0: @param aFs A handle to a file server session. sl@0: @return A pointer to the newly constructed clipboard object. This pointer is sl@0: put onto the cleanup stack. */ sl@0: { sl@0: CClipboard* self=NewLC(aFs); sl@0: self->iStreamDictionary=CStreamDictionary::NewL(); sl@0: // sl@0: TDriveName clipboardDrive = ClipboardFileDrive(); sl@0: TFileName clipboardFile(KClipboardFileName); sl@0: clipboardFile.Insert(0,clipboardDrive); sl@0: sl@0: RFile file; sl@0: TInt r=file.Open(aFs,clipboardFile,EFileRead|EFileWrite); sl@0: sl@0: /** sl@0: * The following branching is desired at this stage: sl@0: * -> if the path is not present, we need to create it and then create the file. sl@0: * -> if the path is present, we only need to create the file sl@0: * -> if both path and file exists, we don't need to do anything sl@0: * sl@0: * Instead of the switch case block we can achieve this by atleast 2 other means. sl@0: * Option 1: sl@0: * RFile file; sl@0: * TInt r=file.Open(aFs,clipboardFile,EFileRead|EFileWrite); sl@0: * if(r == KErrPathNotFound) { sl@0: * BaflUtils::EnsurePathExistsL(aFs,clipboardFile); sl@0: * r=file.Replace(aFs,clipboardFile,EFileRead|EFileWrite); sl@0: * } else if(r == KErrNotFound) { sl@0: * r=file.Replace(aFs,clipboardFile,EFileRead|EFileWrite); sl@0: * } sl@0: * User::LeaveIfError(r); sl@0: * sl@0: * => the above code has extra check to be made when the error is KErrNotFound sl@0: * and has increased code size owing to repeated file.Replace(). sl@0: * sl@0: * Option 2: sl@0: * BaflUtils::EnsurePathExistsL(aFs,clipboardFile); sl@0: * RFile file; sl@0: * TInt r=file.Open(aFs,clipboardFile,EFileRead|EFileWrite); sl@0: * if(r == KErrNotFound) { sl@0: * r=file.Replace(aFs,clipboardFile,EFileRead|EFileWrite); sl@0: * } sl@0: * User::LeaveIfError(r); sl@0: * => Call to BaflUtils::EnsurePathExistsL unconditionally which in turn sl@0: * invokes Rfs::MkDirAll() can be expensive and errorneous. sl@0: * (Check documnentation for RFs::MkDirAll() sl@0: */ sl@0: sl@0: sl@0: //Maintain the order of the switch case and DO NOT add break between the sl@0: //switch cases. They have been written such that only the required set of sl@0: //functionality is executed based on the Error code. sl@0: switch (r) sl@0: { sl@0: case KErrPathNotFound: // path has not been created yet sl@0: BaflUtils::EnsurePathExistsL(aFs,clipboardFile); sl@0: // coverity [fallthrough] sl@0: // Missing break intentional. Read comment before switch case. sl@0: case KErrNotFound: // it does not exist yet sl@0: r=file.Replace(aFs,clipboardFile,EFileRead|EFileWrite); sl@0: // coverity [fallthrough] sl@0: // Missing break intentional. Read comment before switch case. sl@0: default: sl@0: User::LeaveIfError(r); sl@0: } sl@0: // sl@0: CFileStore* store=CDirectFileStore::NewL(file); sl@0: self->iStore=store; sl@0: store->SetTypeL(TUidType(KDirectFileStoreLayoutUid,KClipboardFileUid)); sl@0: return self; sl@0: } sl@0: sl@0: EXPORT_C TInt CClipboard::Clear(RFs& aFs) sl@0: /** Empties the clipboard. sl@0: sl@0: Note that if the file associated with the clipboard's store does not exist, sl@0: this is not regarded as an error and the function completes successfully. sl@0: sl@0: @param aFs A handle to a file server session. sl@0: @return KErrNone if successful, otherwise another of the system-wide error sl@0: codes. */ sl@0: { sl@0: TDriveName clipboardDrive = ClipboardFileDrive(); sl@0: TFileName clipboardFile(KClipboardFileName); sl@0: clipboardFile.Insert(0,clipboardDrive); sl@0: sl@0: RFile file; sl@0: TInt r=file.Open(aFs,clipboardFile,EFileWrite); sl@0: switch (r) sl@0: { sl@0: case KErrNone: // clear without notification sl@0: r=file.SetSize(0); sl@0: file.Close(); sl@0: break; sl@0: case KErrPathNotFound: // no clipboard to clear sl@0: case KErrNotFound: sl@0: r=KErrNone; sl@0: break; sl@0: default: sl@0: r=aFs.Delete(clipboardFile); // just delete the fella sl@0: break; sl@0: } sl@0: return r; sl@0: } sl@0: sl@0: EXPORT_C CClipboard::~CClipboard() sl@0: /** Destructor. Frees all resources owned by the object, prior to its destruction. sl@0: In particular, the file associated with the clipboard's store is closed. sl@0: sl@0: Note that if the clipboard file store has no root stream, i.e. the CommitL() sl@0: member function of CClipboard has not been called prior to deleting the CClipboard sl@0: object, then the file associated with the clipboard file store is deleted. */ sl@0: { sl@0: delete iStreamDictionary; sl@0: CFileStore* store=iStore; sl@0: if (store!=NULL) sl@0: { sl@0: TBool rollback=(store->Root()==KNullStreamId); sl@0: delete store; sl@0: if (rollback) // useless clipboard, no root stream sl@0: { sl@0: TDriveName clipboardDrive = ClipboardFileDrive(); sl@0: TFileName clipboardFile(KClipboardFileName); sl@0: clipboardFile.Insert(0,clipboardDrive); sl@0: sl@0: iFs.Delete(clipboardFile); sl@0: } sl@0: } sl@0: } sl@0: sl@0: EXPORT_C void CClipboard::CommitL() sl@0: /** Commits changes to the clipboard's store. sl@0: sl@0: It externalises the stream dictionary to the clipboard store as the root stream sl@0: and then commits all changes to the store. This function must be called after sl@0: application data has been externalised; failure to do so results in the deletion sl@0: of the file associated with the clipboard store and the loss of data. */ sl@0: { sl@0: __ASSERT_DEBUG(iStore!=NULL&&iStreamDictionary!=NULL,User::Invariant()); sl@0: // sl@0: CFileStore& store=*iStore; sl@0: RStoreWriteStream stream; sl@0: TStreamId root=stream.CreateLC(store); sl@0: stream<< *iStreamDictionary; sl@0: stream.CommitL(); sl@0: CleanupStack::PopAndDestroy(); // dictionary stream sl@0: store.SetRootL(root); sl@0: store.CommitL(); sl@0: } sl@0: sl@0: sl@0: // Copy and paste member functions for low level data types sl@0: sl@0: sl@0: const TUid KClipboardRealTypeUid = { 268435705 }; sl@0: sl@0: sl@0: EXPORT_C void CClipboard::CopyToL(TReal aReal) __SOFTFP sl@0: /** Copies a double-precision floating point value to the clipboard. sl@0: sl@0: Note that the function does not automatically commit changes to the clipboard's sl@0: store. This must be done explicitly. sl@0: sl@0: @param aReal The double-precision floating point value to be copied to the sl@0: clipboard. sl@0: @see CommitL() */ sl@0: { sl@0: __ASSERT_DEBUG(iStore!=NULL&&iStreamDictionary!=NULL,User::Invariant()); sl@0: // sl@0: RStoreWriteStream stream; sl@0: TStreamId id = stream.CreateLC ( Store() ); sl@0: stream << aReal; sl@0: stream.CommitL(); sl@0: CleanupStack::PopAndDestroy(); sl@0: StreamDictionary().AssignL ( KClipboardRealTypeUid, id ); sl@0: } sl@0: sl@0: EXPORT_C TBool CClipboard::PasteFromL ( TReal& aReal ) sl@0: /** Pastes a double-precision floating point value from the clipboard. sl@0: sl@0: If a double-precision floating point value exists on the clipboard, then the sl@0: function restores it to the referenced argument and returns a true value. sl@0: sl@0: If there is no double-precision floating point value on the clipboard, then sl@0: the function returns a false value. The referenced argument is not changed. sl@0: sl@0: @param aReal On return, contains the double-precision floating point value sl@0: found on the clipboard. sl@0: @return ETrue, if a double-precision floating point value exists on the sl@0: clipboard and has been pasted to the referenced argument; EFalse otherwise. */ sl@0: { sl@0: __ASSERT_DEBUG(iStreamDictionary!=NULL,User::Invariant()); sl@0: TStreamId id = StreamDictionary().At(KClipboardRealTypeUid); sl@0: if (id == KNullStreamId) sl@0: return (EFalse); // nothing here sl@0: __ASSERT_DEBUG(iStore!=NULL,User::Invariant()); sl@0: RStoreReadStream stream; sl@0: stream.OpenLC (Store(), id); sl@0: stream >> aReal; sl@0: CleanupStack::PopAndDestroy(); sl@0: return (ETrue); sl@0: } sl@0: sl@0: sl@0: sl@0: // end of BACLIPB.CPP