Update contrib.
1 // Copyright (c) 2005-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // The client-side representation of a WsGraphic artwork
19 #include "../SERVER/w32cmd.h"
23 #include <graphics/WSGRAPHICDRAWERINTERFACE.H>
25 NONSHARABLE_STRUCT(CWsGraphic::CPimpl): public CBase, public MWsClientClass
26 /** @internalComponent @released */
28 friend class CWsGraphic;
31 /** is registered in the graphic manager's array */
33 /** has a peer CWsGraphicDrawer on the server */
36 CPimpl(CWsGraphic& aGraphic);
38 void ConstructL(); //LeaveScan: Member of macroised structure declaration.
40 TInt WriteCreateGraphic(TWsClCmdCreateGraphic& aCreateGraphic,const TDesC8& aData,TInt aWsHandle) const;
44 MWsObjectProvider* iExt;
47 LOCAL_C void MWsGraphicMessageAllocRelease(TAny* aAny)
49 MWsGraphicMessageAlloc::MBuffer* buf = static_cast<MWsGraphicMessageAlloc::MBuffer*>(aAny);
56 NONSHARABLE_CLASS(CWsGraphic::CManager): private CActive, public RWsSession
57 /** Client-side manager singleton for marshalling messages for all CWsGraphics owned by a client
63 static CManager* StaticL(); //LeaveScan: Member of macroised structure declaration.
65 void AddL(CWsGraphic* aGraphic); //LeaveScan: Member of macroised structure declaration.
66 void Replace(CWsGraphic* aGraphic);
67 void Remove(CWsGraphic* aGraphic);
68 CWsGraphic* Find(const TWsGraphicId& aId);
76 void ConstructL(); //LeaveScan: Member of macroised structure declaration.
78 void RunL(); //LeaveScan: Member of macroised structure declaration.
80 static TInt GraphicCompare(const CWsGraphic& aFirst,const CWsGraphic& aSecond);
81 static TInt FlushOnIdle(TAny* aArg);
84 RPointerArray<CWsGraphic> iArray;
86 TBool iFlushScheduled;
89 // TWsGraphicId \\\\\\\\\\\\\\\\\\\\\\\\
91 EXPORT_C TWsGraphicId::TWsGraphicId(TUid aUid):
93 @param aUid UID of the graphic artwork.
94 @publishedAll @released
95 */ iFlags(EWsGraphicIdUid), iId(aUid.iUid)
99 EXPORT_C TWsGraphicId::TWsGraphicId(TInt aId):
100 /** Construct a transient Id
101 @publishedAll @released
102 */ iFlags(EWsGraphicIdTransient), iId(aId)
108 @param aCopy Graphic artwork Id.
110 EXPORT_C TWsGraphicId::TWsGraphicId(const TWsGraphicId& aCopy):
111 iFlags(aCopy.iFlags), iId(aCopy.iId)
115 EXPORT_C TUid TWsGraphicId::Uid() const
117 @return UID of graphic artwork. KNullUid if graphic artwork is transient.
118 @publishedAll @released
122 return TUid::Uid(iId);
127 EXPORT_C TBool TWsGraphicId::IsUid() const
128 /** Identifies whether graphic artwork is non-transient.
129 @return ETrue if graphic artwork is non-transient.
130 @publishedAll @released
132 return (iFlags & EWsGraphicIdUid);
135 EXPORT_C void TWsGraphicId::Set(TUid aUid)
137 @publishedAll @released
140 iFlags = EWsGraphicIdUid;
143 EXPORT_C TInt TWsGraphicId::Id() const
144 /** Returns the transient Id.
145 @return Id of transient graphic artwork. Zero if graphic artwork is non-transient.
146 @publishedAll @released
155 EXPORT_C TBool TWsGraphicId::IsId() const
156 /** Identifies whether graphic artwork is transient.
157 @return ETrue if graphic artwork is transient.
158 @publishedAll @released
160 return (iFlags & EWsGraphicIdTransient);
163 EXPORT_C void TWsGraphicId::Set(TInt aId)
164 /** Set to be a transient Id
165 @publishedAll @released
168 iFlags = EWsGraphicIdTransient;
171 EXPORT_C TInt TWsGraphicId::Compare(const TWsGraphicId& aOther) const
172 /** Compares another Id with this one.
173 @return 0 if the other Id is identical, else -1 if the other Id is to greater than or 1 if the other Id is less than
174 @publishedAll @released
180 else if(iId > aOther.iId)
184 // else we have to compare the iIsUid flag too; again, expect it to be a match 99.99% of these times
185 else if(IsUid() == aOther.IsUid())
189 // collisions of id but not iIsUid are going to be really really rare
200 // CWsGraphicManager \\\\\\\\\\\\\\\\\\\\\\\\
202 CWsGraphic::CManager* CWsGraphic::CManager::StaticL()
204 CManager* singleton = RWsBuffer::WsGraphicManager();
207 singleton = new(ELeave) CManager;
208 CleanupStack::PushL(singleton);
209 singleton->ConstructL();
210 CleanupStack::Pop(singleton);
211 __ASSERT_DEBUG(singleton == RWsBuffer::WsGraphicManager(),Panic(EW32PanicGraphicInternal));
216 CWsGraphic::CManager::~CManager()
218 __ASSERT_DEBUG(!ResourceCount(),Panic(EW32PanicGraphicInternal));
219 __ASSERT_DEBUG(!iArray.Count(),Panic(EW32PanicGraphicOrphaned));
221 __ASSERT_DEBUG(!RWsBuffer::WsGraphicManager(),Panic(EW32PanicGraphicInternal));
226 CWsGraphic::CManager::CManager(): CActive(CActive::EPriorityStandard)
230 void CWsGraphic::CManager::ConstructL()
232 User::LeaveIfError(Connect());
233 iBuffer->SetWsGraphicManager(this);
234 CActiveScheduler::Add(this);
235 // coverity[negative_returns]
236 iFlusher = CIdle::NewL(CActive::EPriorityIdle);
239 void CWsGraphic::CManager::Inc()
244 void CWsGraphic::CManager::Dec()
252 RWsBuffer* CWsGraphic::CManager::Buffer()
257 // used by CWsGraphic
259 void CWsGraphic::CManager::AddL(CWsGraphic* aGraphic)
260 /** Leaves if the graphic couldn't be added to the list
261 @internalComponent @released */
263 __ASSERT_ALWAYS(aGraphic && aGraphic->iPimpl,Panic(EW32PanicGraphicInternal));
264 __ASSERT_ALWAYS(aGraphic->Id().IsId() || aGraphic->Id().IsUid(),Panic(EW32PanicGraphicInternal));
265 __ASSERT_ALWAYS(!(aGraphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered),Panic(EW32PanicGraphicInternal));
266 iArray.InsertInOrderL(aGraphic,GraphicCompare);
267 __ASSERT_ALWAYS(0 <= iArray.FindInOrder(aGraphic,GraphicCompare),Panic(EW32PanicGraphicInternal));
268 aGraphic->iPimpl->iFlags |= CWsGraphic::CPimpl::ERegistered;
272 void CWsGraphic::CManager::Replace(CWsGraphic* aGraphic)
273 /** @internalComponent @released */
275 __ASSERT_ALWAYS(aGraphic && aGraphic->iPimpl,Panic(EW32PanicGraphicInternal));
276 __ASSERT_ALWAYS(!(aGraphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered),Panic(EW32PanicGraphicInternal));
277 __ASSERT_ALWAYS(aGraphic->Id().IsId() || aGraphic->Id().IsUid(),Panic(EW32PanicGraphicInternal));
278 const TInt idx = iArray.FindInOrder(aGraphic,GraphicCompare);
279 __ASSERT_ALWAYS(0 <= idx,Panic(EW32PanicGraphicInternal));
280 __ASSERT_ALWAYS(iArray[idx]->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered,Panic(EW32PanicGraphicInternal));
281 iArray[idx]->iPimpl->iFlags &= ~CWsGraphic::CPimpl::ERegistered;
282 iArray[idx] = aGraphic;
283 iArray[idx]->iPimpl->iFlags |= CWsGraphic::CPimpl::ERegistered;
287 void CWsGraphic::CManager::Remove(CWsGraphic* aGraphic)
288 /** @internalComponent @released */
290 __ASSERT_ALWAYS(aGraphic && aGraphic->iPimpl,Panic(EW32PanicGraphicInternal));
291 __ASSERT_ALWAYS(aGraphic->Id().IsId() || aGraphic->Id().IsUid(),Panic(EW32PanicGraphicInternal));
292 __ASSERT_ALWAYS(aGraphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered,Panic(EW32PanicGraphicInternal));
293 const TInt idx = iArray.FindInOrder(aGraphic,GraphicCompare);
294 __ASSERT_ALWAYS(0 <= idx,Panic(EW32PanicGraphicInternal));
295 iArray[idx]->iPimpl->iFlags &= ~CWsGraphic::CPimpl::ERegistered;
303 CWsGraphic* CWsGraphic::CManager::Find(const TWsGraphicId& aId)
304 /** Find the active artwork identified by the Id
305 @return NULL if no active artwork has the Id
306 @publishedAll @released */
308 /* RPointerArray can only FindInOrder other T*, which is a needless shame. Therefore this search is
310 const TInt count = iArray.Count();
311 for(TInt i=0; i<count; i++)
313 CWsGraphic* graphic = iArray[i];
314 __ASSERT_ALWAYS(graphic && graphic->iPimpl && (graphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered),Panic(EW32PanicGraphicInternal));
315 const TWsGraphicId& candidate = graphic->Id();
316 if(0 == candidate.Compare(aId))
319 __ASSERT_DEBUG(i == iArray.FindInOrder(graphic,GraphicCompare),Panic(EW32PanicGraphicInternal));
327 void CWsGraphic::CManager::Queue()
331 GraphicMessageReady(&iStatus);
336 void CWsGraphic::CManager::RunL()
338 if(0 < iStatus.Int())
340 // a message to fetch!
341 const TInt msgLen = iStatus.Int();
343 MWsGraphicMessageAlloc* allocator = NULL;
344 MWsGraphicMessageAlloc::MBuffer* theirBuf = NULL;
345 HBufC8* ourBuf = HBufC8::New(msgLen);
348 CleanupStack::PushL(ourBuf);
349 buf.Set(ourBuf->Des());
351 else // try to see if the destination CWsGraphic can allocate for us
353 const TInt handle = GraphicFetchHeaderMessage();
354 CWsGraphic* dest = reinterpret_cast<CWsGraphic*>(handle & ~0x03);
355 // check if it's valid
356 if(KErrNotFound != iArray.Find(dest) && dest->iPimpl && dest->iPimpl->iExt)
358 // check if the client is able to alloc memory for us
359 allocator = dest->iPimpl->iExt->ObjectInterface<MWsGraphicMessageAlloc>();
363 theirBuf = allocator->Alloc(msgLen);
366 CleanupStack::PushL(TCleanupItem(MWsGraphicMessageAllocRelease,theirBuf));
367 buf.Set(theirBuf->Buffer());
372 if(!ourBuf && !theirBuf)
374 GraphicMessageCancel();
375 GraphicAbortMessage(KErrNoMemory);
379 GetGraphicMessage(buf);
380 // decode header and body
381 RDesReadStream in(buf);
383 const TInt header = in.ReadInt32L();
384 __ASSERT_COMPILE(sizeof(header) == sizeof(TInt32));
385 const TInt clientHandle = (header & ~0x03);
386 const TInt msgType = (header & 0x03);
387 const TPtr8 body = buf.MidTPtr(sizeof(header));
389 CWsGraphic* dest = (CWsGraphic*)clientHandle;
390 if(KErrNotFound != iArray.Find(dest))
394 case EWsGraphMessageTypeUser:
395 dest->HandleMessage(body);
398 Panic(EW32PanicGraphicInternal);
406 CleanupStack::PopAndDestroy(ourBuf);
410 CleanupStack::PopAndDestroy(theirBuf);
412 // done, wait for next message
417 void CWsGraphic::CManager::DoCancel()
419 GraphicMessageCancel();
422 TInt CWsGraphic::CManager::GraphicCompare(const CWsGraphic& aFirst,const CWsGraphic& aSecond)
423 /** Compares two graphics for id equality
427 return aFirst.Id().Compare(aSecond.Id());
430 void CWsGraphic::CManager::ScheduleFlush()
431 /** Request to schedule flush when idle
439 iFlushScheduled = ETrue;
440 iFlusher->Start(TCallBack(CWsGraphic::CManager::FlushOnIdle,this));
443 TInt CWsGraphic::CManager::FlushOnIdle(TAny* aArg)
444 /** Flush buffer when idle and there is outstanding data in it
449 CWsGraphic::CManager* mgr = reinterpret_cast<CWsGraphic::CManager*>(aArg);
452 mgr->iFlushScheduled = EFalse;
453 if (mgr->iBuffer && !mgr->iBuffer->IsEmpty())
454 mgr->iBuffer->Flush();
457 return 0; // complete
460 // CWsGraphic::CPimpl \\\\\\\\\\\\\\\\\\\\\\\\
462 CWsGraphic::CPimpl::CPimpl(CWsGraphic& aGraphic):
463 iGraphic(aGraphic), iId(TWsGraphicId::EUninitialized)
467 CWsGraphic::CPimpl::~CPimpl()
475 void CWsGraphic::CPimpl::ConstructL()
477 iManager = CManager::StaticL();
479 iBuffer = iManager->Buffer();
482 TInt CWsGraphic::CPimpl::WriteCreateGraphic(TWsClCmdCreateGraphic& aCreateGraphic,const TDesC8& aData,TInt aWsHandle) const
483 /** Writes the CreateGraphic message to the server. If the data will not fit in the buffer, uses remote-read
484 @internalComponent @released */
486 aCreateGraphic.iDataLen = aData.Size();
487 aCreateGraphic.iRemoteReadData = ((aData.Size()+sizeof(aCreateGraphic))>(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)));
488 if(aCreateGraphic.iRemoteReadData)
490 return iBuffer->WriteReplyByProvidingRemoteReadAccess(aWsHandle,EWsClOpCreateGraphic,&aCreateGraphic,sizeof(aCreateGraphic),&aData);
492 else if(aCreateGraphic.iDataLen)
494 return iBuffer->WriteReplyWs(&aCreateGraphic,sizeof(aCreateGraphic),aData.Ptr(),aData.Size(),EWsClOpCreateGraphic);
498 return iBuffer->WriteReplyWs(&aCreateGraphic,sizeof(aCreateGraphic),EWsClOpCreateGraphic);
502 // CWsGraphic \\\\\\\\\\\\\\\\\\\\\\\\
507 EXPORT_C CWsGraphic::CWsGraphic()
514 EXPORT_C CWsGraphic::~CWsGraphic()
521 Completes construction of the baseclass. All the overloaded BaseConstructL() methods
522 should invoke this method to complete the construction of the baseclass.
524 void CWsGraphic::BaseConstructL()
526 iPimpl = new(ELeave) CPimpl(*this);
527 iPimpl->ConstructL();
530 EXPORT_C void CWsGraphic::BaseConstructL(TUid aUid,TUid aType,const TDesC8& aData)
532 Constructs a piece of non-transient graphic artwork.
534 @param aUid Graphic artwork UID.
535 @param aType Graphic artwork type.
536 @param aData User specific data.
540 TWsClCmdCreateGraphic createGraphic;
541 createGraphic.iFlags = EWsGraphicIdUid;
542 createGraphic.iId = aUid.iUid;
543 createGraphic.iType = aType;
544 createGraphic.iClientHandle = (TInt)this;
546 TInt ret = iPimpl->WriteCreateGraphic(createGraphic,aData,iPimpl->iManager->WsHandle());
547 User::LeaveIfError(ret);
549 iPimpl->iWsHandle=ret;
551 iPimpl->iFlags = CPimpl::EHasPeer;
552 iPimpl->iManager->AddL(this);
555 EXPORT_C void CWsGraphic::BaseConstructL(TUid aType,const TDesC8& aData)
557 Constructs a piece of transient graphic artwork.
558 @param aType Graphic artwork type.
559 @param aData User specific data.
564 TWsClCmdCreateGraphic createGraphic;
565 createGraphic.iFlags = EWsGraphicIdTransient;
566 createGraphic.iId = 0;
567 createGraphic.iType = aType;
568 createGraphic.iClientHandle = (TInt)this;
570 TInt ret = iPimpl->WriteCreateGraphic(createGraphic,aData,iPimpl->iManager->WsHandle());
571 User::LeaveIfError(ret);
573 iPimpl->iWsHandle = ret;
575 // fetch id from server
576 TPckgBuf<TWsClCmdGdGetId> cmd;
577 User::LeaveIfError(iPimpl->WriteReplyP(TWriteDescriptorType(&cmd),EWsGdOpGetGraphicId));
580 __DEBUG_ONLY(Panic(EW32PanicGraphicInternal));
581 User::Leave(KErrGeneral);
583 iPimpl->iId = cmd().iId;
585 iPimpl->iFlags = CPimpl::EHasPeer;
586 iPimpl->iManager->AddL(this);
589 EXPORT_C void CWsGraphic::BaseConstructL(const TWsGraphicId& aReplace,TUid aType,const TDesC8& aData)
591 Atomically replace the artwork that already exists with this artwork.
592 If failure to properly construct the replacement artwork occurs, the replacee artwork will remain
593 @param aReplace Graphic artwork which will be replaced.
594 @param aType New graphic artwork type.
595 @param aData User specific data.
600 CWsGraphic* dup = iPimpl->iManager->Find(aReplace);
601 if(!dup || !dup->iPimpl)
603 Panic(EW32PanicGraphicInternal);
606 const TUint flags = aReplace.IsUid()?EWsGraphicIdUid:EWsGraphicIdTransient;
608 TWsClCmdCreateGraphic createGraphic;
609 createGraphic.iFlags = EWsGraphicReplace|flags;
610 createGraphic.iId = aReplace.IsUid()?aReplace.Uid().iUid:aReplace.Id();
611 createGraphic.iType = aType;
612 createGraphic.iClientHandle = (TInt)this;
614 TInt ret = iPimpl->WriteCreateGraphic(createGraphic,aData,iPimpl->iManager->WsHandle());
620 iPimpl->iWsHandle = ret;
621 iPimpl->iId = aReplace;
623 iPimpl->iFlags = CPimpl::EHasPeer;
624 iPimpl->iManager->Replace(this);
626 // when WriteCreateGraphic succeeds, the replacee drawer has already been destroyed serverside
627 // so this cleanup is not quite the same as Destroy(), as it doesn't free server side
628 // coverity[var_deref_op]
629 dup->iPimpl->iWsHandle = 0;
631 dup->iPimpl->iFlags &= ~CPimpl::EHasPeer;
635 Shares the graphic artwork with all the client sessions.
636 Sharing globally trumps explicit shares.
637 @return KErrNone if the graphic is globally shared, else one of the system-wide error codes.
639 EXPORT_C TInt CWsGraphic::ShareGlobally()
645 return iPimpl->WriteReply(EWsGdOpShareGlobally);
649 Prevents this graphic artwork from being shared with all the client sessions.
650 A graphic artwork that isn't shared explicitly is only available to clients it
651 has been explicitly shared with using Share().
652 @return KErrNone if the graphic is not globally shared, else one of the system-wide error codes.
654 EXPORT_C TInt CWsGraphic::UnShareGlobally()
660 return iPimpl->WriteReply(EWsGdOpUnShareGlobally);
664 Explicitly shares this graphic artwork with client sessions with the specified Secure ID.
665 @param aClientId the Secure ID of the client sessions to share with.
666 @return KErrNone If the graphic artwork was shared, else one of the system-wide error codes.
668 EXPORT_C TInt CWsGraphic::Share(TSecureId aClientId)
674 return iPimpl->WriteReply(&aClientId,sizeof(TSecureId),EWsGdOpShare);
678 Stops this graphic artwork from being shared with all client sessions with the specific Secure ID.
679 ShareGlobally() trumps explicit sharing.
680 @param aClientId the Secure ID of the client sessions to not share with
681 @return KErrNone if the graphic artwork is no longer shared, KErrNotFound if the graphic was not shared anyway, else one of the system-wide error codes
683 EXPORT_C TInt CWsGraphic::UnShare(TSecureId aClientId)
689 return iPimpl->WriteReply(&aClientId,sizeof(TSecureId),EWsGdOpUnShare);
693 Returns graphic artwork Id.
694 @return Graphic artwork Id. KNullWsGraphicId if graphic artwork is not active.
696 EXPORT_C const TWsGraphicId& CWsGraphic::Id() const
705 static const TInt KNullWsGraphicId[4] = //binary compatible with TWsGraphicId
709 __ASSERT_COMPILE(sizeof(KNullWsGraphicId) == sizeof(TWsGraphicId));
710 return reinterpret_cast<const TWsGraphicId&>(KNullWsGraphicId);
715 Checks whether a peer of this graphic artwork has been fully constructed on the server.
716 @return ETrue if this graphic artwork has a peer CWsGraphic on the server.
718 EXPORT_C TBool CWsGraphic::IsActive() const
720 return (iPimpl && iPimpl->iWsHandle && (iPimpl->iFlags & CPimpl::EHasPeer));
724 Derived class can override this method to provide custom operations when the client is closed.
726 EXPORT_C void CWsGraphic::OnClientClose()
731 Sends message to this graphic artwork peer on the server.
732 @param aData User specific data.
734 EXPORT_C void CWsGraphic::SendMessage(const TDesC8& aData) const
736 TWsClCmdGdSendMessage cmd;
737 cmd.iDataLen = aData.Size();
738 __ASSERT_DEBUG(cmd.iDataLen, Panic(EW32PanicGraphicNullData));
739 cmd.iRemoteReadData = ((aData.Size()+sizeof(cmd))>(TInt)(iPimpl->iBuffer->BufferSize()-sizeof(TWsCmdHeader)));
740 if(cmd.iRemoteReadData)
742 iPimpl->WriteReplyByProvidingRemoteReadAccess(&cmd,sizeof(cmd),&aData,EWsGdOpSendMsg);
743 //ignore return value!
747 iPimpl->Write(&cmd,sizeof(cmd),aData.Ptr(),aData.Size(),EWsGdOpSendMsg);
751 EXPORT_C TInt CWsGraphic::SendSynchronMessage(const TDesC8& aData) const
753 TWsClCmdGdSendMessage cmd;
754 cmd.iDataLen = aData.Size();
755 __ASSERT_DEBUG(cmd.iDataLen, Panic(EW32PanicGraphicNullData));
756 cmd.iRemoteReadData = ((aData.Size()+sizeof(cmd))>(TInt)(iPimpl->iBuffer->BufferSize()-sizeof(TWsCmdHeader)));
757 if(cmd.iRemoteReadData)
759 return iPimpl->WriteReplyByProvidingRemoteReadAccess(&cmd,sizeof(cmd),&aData,EWsGdOpSendSynchronMsg);
763 return iPimpl->WriteReply(&cmd,sizeof(cmd),aData.Ptr(),aData.Size(),EWsGdOpSendSynchronMsg);
768 Flushes window server command buffer
769 @return One of system-wide error codes.
771 EXPORT_C TInt CWsGraphic::Flush() const
773 return iPimpl->iBuffer->Flush();
776 EXPORT_C void CWsGraphic::Destroy()
777 /** Destroys the corresponding CWsGraphicDrawer instance on the server
781 if(iPimpl && (iPimpl->iFlags & CPimpl::ERegistered))
783 iPimpl->iManager->Remove(this);
785 if(IsActive()) // if iPimpl==NULL, IsActive() returns false
787 __ASSERT_DEBUG(iPimpl != NULL, Panic(EW32PanicGraphicInternal));
789 iPimpl->Write(EWsGdOpFree);
790 iPimpl->iWsHandle = 0;
791 iPimpl->iFlags &= ~CPimpl::EHasPeer;
792 iPimpl->iManager->ScheduleFlush();
796 EXPORT_C void CWsGraphic::SetGraphicExtension(MWsObjectProvider* aExt)
800 EXPORT_C RWsSession& CWsGraphic::Session()
802 return *iPimpl->iManager;
805 EXPORT_C TInt CWsGraphic::CWsGraphic_Reserved1()
807 return KErrNotSupported;
810 EXPORT_C TInt CWsGraphic::CWsGraphic_Reserved2()
812 return KErrNotSupported;
815 EXPORT_C TInt CWsGraphic::CWsGraphic_Reserved3()
817 return KErrNotSupported;
820 // TWsGraphicMsgFixedBase \\\\\\\\\\\\\\\\\\\\\\\\
822 EXPORT_C TWsGraphicMsgFixedBase::TWsGraphicMsgFixedBase(TUid aTypeId,TInt aSizeOfDerived):
823 /** Protected constructor for subclasses to call
824 @param aTypeId The UID representing this type of data
825 @param aSizeOf The size of the derived class
829 TMyDerivedFixedMsg::TMyDerivedFixedMsg(): TWsGraphicMsgFixedBase(KUidMyDerivedType,sizeof(TMyDerivedFixedMsg)), ...
831 */ iTypeId(aTypeId), iSize(aSizeOfDerived-sizeof(TWsGraphicMsgFixedBase))
833 __ASSERT_COMPILE(sizeof(*this) == (sizeof(TInt32)*2));
836 EXPORT_C TPtrC8 TWsGraphicMsgFixedBase::Pckg() const
837 /** @return this fixed message as a descriptor so that it can be passed as draw data in the CWindowGc::DrawWsGraphic command directly if only one such message is to be sent
839 return TPtrC8(reinterpret_cast<const TUint8*>(this),sizeof(*this) + iSize);
842 EXPORT_C TUid TWsGraphicMsgFixedBase::TypeId() const
843 /** @return the UID identifying the type of the data that follows */
848 EXPORT_C TInt TWsGraphicMsgFixedBase::Size() const
849 /** @return the size of the derived class (not including this fixed base class size) */
854 // RWsGraphicMsgBuf \\\\\\\\\\\\\\\\\\\\\\\\
856 EXPORT_C RWsGraphicMsgBuf::RWsGraphicMsgBuf()
857 /** Default Constructor */
861 TInt RWsGraphicMsgBuf::IntAt(TInt aOfs) const
862 /** @internalComponent @released */
864 if((aOfs < 0) || ((aOfs+sizeof(TInt)) > Length()))
866 Panic(EW32PanicGraphicBadBuffer);
869 memcpy(&ret,Ptr()+aOfs,sizeof(TInt));
873 EXPORT_C TInt RWsGraphicMsgBuf::Append(TUid aTypeId,const TDesC8& aData)
874 /** Append a descriptor as data
875 @param aTypeId the type of the message to append
876 @param aData arbitrary length data consisting of the whole message
877 @return KErrNone if successful, else a system-wide error code
879 TInt err = ExpandForAppend(aData.Length());
884 WriteHeader(aTypeId,aData.Length());
886 Insert(Length(),aData);
890 EXPORT_C TInt RWsGraphicMsgBuf::Append(TUid aTypeId,const TDesC16& aData)
891 /** Append a descriptor as data
892 @param aTypeId the type of the message to append
893 @param aData arbitrary length data consisting of the whole message
894 @return KErrNone if successful, else a system-wide error code
897 TInt err = Append(aTypeId,aData.Size(),data);
903 data.Copy(reinterpret_cast<const TUint8*>(aData.Ptr()),aData.Size());
907 TInt RWsGraphicMsgBuf::ExpandForAppend(TInt aDataLen)
908 /** @internalComponent @released */
910 __ASSERT_COMPILE(sizeof(TInt) == sizeof(TInt32));
911 const TInt required = (sizeof(TUid) + sizeof(TInt) + aDataLen);
912 if(MaxLength() < (Length() + required))
914 TInt err = ReAlloc(Length()+required);
923 void RWsGraphicMsgBuf::WriteHeader(TUid aTypeId,TInt aLen)
924 /** @internalComponent @released */
926 __ASSERT_COMPILE(sizeof(TInt) == sizeof(TInt32));
928 TPckgBuf<TInt32> i(aTypeId.iUid);
934 EXPORT_C TInt RWsGraphicMsgBuf::Append(TUid aTypeId,TInt aLen,TPtr8& aPtr)
935 /** Append a message of the specified length and type, and return a pointer to
936 allow client code to modify the message.
937 aPtr is only set if the append succeeds.
938 aPtr is only valid until the next message is appended to the buffer.
939 @param aTypeId the type of the message to append
940 @param aLen the length of the message to be reserved
941 @param aPtr a modifiable descriptor to be used by the client code to write into the message body
942 @return KErrNone if successful, else a system-wide error code
944 TInt err = ExpandForAppend(aLen);
949 WriteHeader(aTypeId,aLen);
951 TInt usedLen = Length();
952 SetLength(usedLen+aLen);
953 aPtr.Set(MidTPtr(usedLen,aLen));
954 aPtr.Fill('_'); // prettier when debugging, but want consistant behaviour in release builds
959 EXPORT_C void RWsGraphicMsgBuf::Remove(TInt aIndex)
960 /** Remove a message from the buffer
961 @panic if the index is out of bounds
962 @param aIndex the ordinal position of message to be removed
964 if((aIndex < 0) || (aIndex >= Count()))
966 Panic(EW32PanicGraphicBadBuffer);
968 TPtrC8 ptr = Data(aIndex);
969 const TInt ofs = (ptr.Ptr()-(sizeof(TInt)*2)-Ptr());
970 Delete(ofs,ptr.Length() + (sizeof(TInt)*2));
973 EXPORT_C TInt RWsGraphicMsgBuf::Count() const
974 /** Returns the number of messages in the buffer
975 @return the number of messages in the buffer
977 const TInt length = Length();
978 TInt ofs = 0, count = 0;
982 ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
986 Panic(EW32PanicGraphicBadBuffer);
991 EXPORT_C TUid RWsGraphicMsgBuf::TypeId(TInt aIndex) const
992 /** Returns the Type ID of a message in the buffer
993 @param aIndex the ordinal position of the message
994 @return the Type ID of the message
995 @panic if the index is out of bounds
997 const TInt length = Length();
998 TInt ofs = 0, count = 0;
1003 return TUid::Uid(IntAt(ofs));
1006 ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
1008 Panic(EW32PanicGraphicBadBuffer);
1009 return KNullUid; //dumb compiler
1012 EXPORT_C TPtrC8 RWsGraphicMsgBuf::Data(TInt aIndex) const
1013 /** Returns a non-modifiable descriptor of a message body in the buffer
1014 @param aIndex the ordinal position of the message
1015 @return the message body
1016 @panic if the index is out of bounds
1018 const TInt length = Length();
1019 TInt ofs = 0, count = 0;
1024 return Mid(ofs+(sizeof(TInt)*2),IntAt(ofs+sizeof(TInt)));
1027 ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
1029 Panic(EW32PanicGraphicBadBuffer);
1030 return TPtrC8(KNullDesC8()); //dumb compiler
1033 EXPORT_C TPtr8 RWsGraphicMsgBuf::Data(TInt aIndex)
1034 /** Returns a modifiable descriptor of a message body in the buffer
1035 The returned TPtr8 is only valid until the next message is appended to the buffer.
1036 @param aIndex the ordinal position of the message
1037 @return the message body
1038 @panic if the index is out of bounds
1040 const TInt length = Length();
1041 TInt ofs = 0, count = 0;
1046 return MidTPtr(ofs+(sizeof(TInt)*2),IntAt(ofs+sizeof(TInt)));
1049 ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
1051 Panic(EW32PanicGraphicBadBuffer);
1052 return TPtr8(NULL,0); //dumb compiler
1055 EXPORT_C const TDesC8& RWsGraphicMsgBuf::Pckg() const
1056 /** Returns the message buffer as a descriptor. Example:
1058 RWsGraphicMsgBuf msgBuf;
1061 TWsGraphicId id(...);
1062 SystemGc().DrawWsGraphic(id,Rect(),msgBuf.Pckg());
1065 @see CWindowGc::DrawWsGraphic
1066 @see CCoeControl::Draw
1067 @return the message buffer to be attached a command to draw a CWsGraphic
1072 EXPORT_C TInt RWsGraphicMsgBuf::Append(const TWsGraphicMsgFixedBase& aMsg)
1073 /** Append a fixed-size message
1074 @param aMsg the fixed-size message to append
1075 @return KErrNone if successful, else a system-wide error code
1077 __ASSERT_COMPILE(sizeof(TWsGraphicMsgFixedBase) == (sizeof(TInt)*2));
1078 TInt err = ExpandForAppend(aMsg.Size());
1084 RBuf8::Append(reinterpret_cast<const TUint8*>(&aMsg),sizeof(TWsGraphicMsgFixedBase) + aMsg.Size());
1089 EXPORT_C void RWsGraphicMsgBuf::GetFixedMsg(TWsGraphicMsgFixedBase& aMsg,TInt aIndex) const
1090 /** Returns a copy of a fixed-size message in the buffer
1091 @param a copy of the message
1092 @param aIndex the ordinal position of the message
1093 @panic if the index is out of bounds
1094 @panic the message specified is not of the correct type
1096 __ASSERT_COMPILE(sizeof(TWsGraphicMsgFixedBase) == (sizeof(TInt32)*2));
1097 const TInt KHeaderSize = sizeof(TWsGraphicMsgFixedBase);
1098 const TInt length = Length();
1099 TInt ofs = 0, count = 0;
1104 if((TUid::Uid(IntAt(ofs)) != aMsg.TypeId()) ||
1105 (IntAt(ofs+sizeof(TInt)) != aMsg.Size()) ||
1106 ((ofs + KHeaderSize + aMsg.Size()) > length))
1108 Panic(EW32PanicGraphicBadBuffer);
1110 memcpy(&aMsg,(Ptr()+ofs),KHeaderSize + aMsg.Size());
1114 ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
1116 Panic(EW32PanicGraphicBadBuffer);