os/graphics/windowing/windowserver/nonnga/CLIENT/WSGRAPHIC.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) 2005-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
// The client-side representation of a WsGraphic artwork
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include <w32std.h>
sl@0
    19
#include "../SERVER/w32cmd.h"
sl@0
    20
#include "w32comm.h"
sl@0
    21
#include "CLIENT.H"
sl@0
    22
#include <s32mem.h>
sl@0
    23
#include <graphics/wsgraphicdrawerinterface.h>
sl@0
    24
/** Panics the client. This will result in the client thread being destroyed. */
sl@0
    25
GLREF_C void Panic(TW32Panic aPanic);
sl@0
    26
sl@0
    27
NONSHARABLE_STRUCT(CWsGraphic::CPimpl): public CBase, public MWsClientClass
sl@0
    28
/** @internalComponent @released */
sl@0
    29
	{
sl@0
    30
	friend class CWsGraphic;
sl@0
    31
	enum
sl@0
    32
		{
sl@0
    33
		/** is registered in the graphic manager's array */
sl@0
    34
		ERegistered = 0x01,
sl@0
    35
		/** has a peer CWsGraphicDrawer on the server */
sl@0
    36
		EHasPeer	= 0x02
sl@0
    37
		};
sl@0
    38
	CPimpl(CWsGraphic& aGraphic);
sl@0
    39
	~CPimpl();
sl@0
    40
	void ConstructL();		//LeaveScan: Member of macroised structure declaration.
sl@0
    41
	CManager* iManager;
sl@0
    42
	TInt WriteCreateGraphic(TWsClCmdCreateGraphic& aCreateGraphic,const TDesC8& aData,TInt aWsHandle) const;
sl@0
    43
	CWsGraphic& iGraphic;
sl@0
    44
	TWsGraphicId iId;
sl@0
    45
	TUint iFlags;
sl@0
    46
	MWsObjectProvider* iExt;
sl@0
    47
	};
sl@0
    48
sl@0
    49
LOCAL_C void MWsGraphicMessageAllocRelease(TAny* aAny)
sl@0
    50
	{
sl@0
    51
	MWsGraphicMessageAlloc::MBuffer* buf = static_cast<MWsGraphicMessageAlloc::MBuffer*>(aAny);
sl@0
    52
	if(buf)
sl@0
    53
		{
sl@0
    54
		buf->Release();
sl@0
    55
		}
sl@0
    56
	}
sl@0
    57
sl@0
    58
NONSHARABLE_CLASS(CWsGraphic::CManager): private CActive, public RWsSession
sl@0
    59
/** Client-side manager singleton for marshalling messages for all CWsGraphics owned by a client
sl@0
    60
sl@0
    61
@publishedAll
sl@0
    62
@released
sl@0
    63
*/	{
sl@0
    64
public:
sl@0
    65
	static CManager* StaticL();			  //LeaveScan:  Member of macroised structure declaration.
sl@0
    66
	// used by CWsGraphic
sl@0
    67
	void AddL(CWsGraphic* aGraphic);	  //LeaveScan:  Member of macroised structure declaration.
sl@0
    68
	void Replace(CWsGraphic* aGraphic);
sl@0
    69
	void Remove(CWsGraphic* aGraphic);
sl@0
    70
	CWsGraphic* Find(const TWsGraphicId& aId);
sl@0
    71
	void Inc();
sl@0
    72
	void Dec();
sl@0
    73
	RWsBuffer* Buffer();
sl@0
    74
	void ScheduleFlush();
sl@0
    75
private:
sl@0
    76
	CManager();
sl@0
    77
	~CManager();
sl@0
    78
	void ConstructL();					 //LeaveScan:  Member of macroised structure declaration.
sl@0
    79
	void Queue();
sl@0
    80
	void RunL();						 //LeaveScan:  Member of macroised structure declaration.
sl@0
    81
	void DoCancel();
sl@0
    82
	static TInt GraphicCompare(const CWsGraphic& aFirst,const CWsGraphic& aSecond);
sl@0
    83
	static TInt FlushOnIdle(TAny* aArg);
sl@0
    84
private:
sl@0
    85
	TInt iRefCount;
sl@0
    86
	RPointerArray<CWsGraphic> iArray;
sl@0
    87
	CIdle* iFlusher;
sl@0
    88
	TBool iFlushScheduled;
sl@0
    89
	};
sl@0
    90
sl@0
    91
// TWsGraphicId \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
    92
sl@0
    93
EXPORT_C TWsGraphicId::TWsGraphicId(TUid aUid):
sl@0
    94
/** Construct a UID
sl@0
    95
    @param aUid UID of the graphic artwork.
sl@0
    96
    @publishedAll @released
sl@0
    97
*/	iFlags(EWsGraphicIdUid), iId(aUid.iUid)
sl@0
    98
	{
sl@0
    99
	}
sl@0
   100
sl@0
   101
EXPORT_C TWsGraphicId::TWsGraphicId(TInt aId):
sl@0
   102
/** Construct a transient Id
sl@0
   103
	@publishedAll @released
sl@0
   104
*/	iFlags(EWsGraphicIdTransient), iId(aId)
sl@0
   105
	{
sl@0
   106
	}
sl@0
   107
sl@0
   108
/** 
sl@0
   109
Copy constructor.
sl@0
   110
@param aCopy Graphic artwork Id.
sl@0
   111
*/
sl@0
   112
EXPORT_C TWsGraphicId::TWsGraphicId(const TWsGraphicId& aCopy):
sl@0
   113
	iFlags(aCopy.iFlags), iId(aCopy.iId)
sl@0
   114
	{
sl@0
   115
	}
sl@0
   116
sl@0
   117
EXPORT_C TUid TWsGraphicId::Uid() const
sl@0
   118
/** Returns UID.
sl@0
   119
    @return UID of graphic artwork. KNullUid if graphic artwork is transient.
sl@0
   120
    @publishedAll @released
sl@0
   121
*/	{
sl@0
   122
	if(IsUid())
sl@0
   123
		{
sl@0
   124
		return TUid::Uid(iId);
sl@0
   125
		}
sl@0
   126
	return KNullUid;
sl@0
   127
	}
sl@0
   128
sl@0
   129
EXPORT_C TBool TWsGraphicId::IsUid() const
sl@0
   130
/** Identifies whether graphic artwork is non-transient.
sl@0
   131
    @return ETrue if graphic artwork is non-transient.
sl@0
   132
    @publishedAll @released
sl@0
   133
*/	{
sl@0
   134
	return (iFlags & EWsGraphicIdUid);
sl@0
   135
	}
sl@0
   136
sl@0
   137
EXPORT_C void TWsGraphicId::Set(TUid aUid)
sl@0
   138
/** Set to be a UID
sl@0
   139
	@publishedAll @released
sl@0
   140
*/	{
sl@0
   141
	iId = aUid.iUid;
sl@0
   142
	iFlags = EWsGraphicIdUid;
sl@0
   143
	}
sl@0
   144
sl@0
   145
EXPORT_C TInt TWsGraphicId::Id() const
sl@0
   146
/** Returns the transient Id.
sl@0
   147
    @return Id of transient graphic artwork. Zero if graphic artwork is non-transient.
sl@0
   148
	@publishedAll @released
sl@0
   149
*/	{
sl@0
   150
	if(IsId())
sl@0
   151
		{
sl@0
   152
		return iId;
sl@0
   153
		}
sl@0
   154
	return 0;
sl@0
   155
	}
sl@0
   156
sl@0
   157
EXPORT_C TBool TWsGraphicId::IsId() const
sl@0
   158
/** Identifies whether graphic artwork is transient.
sl@0
   159
    @return ETrue if graphic artwork is transient.
sl@0
   160
    @publishedAll @released
sl@0
   161
*/	{
sl@0
   162
	return (iFlags & EWsGraphicIdTransient);
sl@0
   163
	}
sl@0
   164
sl@0
   165
EXPORT_C void TWsGraphicId::Set(TInt aId)
sl@0
   166
/** Set to be a transient Id
sl@0
   167
	@publishedAll @released
sl@0
   168
*/	{
sl@0
   169
	iId = aId;
sl@0
   170
	iFlags = EWsGraphicIdTransient;
sl@0
   171
	}
sl@0
   172
sl@0
   173
EXPORT_C TInt TWsGraphicId::Compare(const TWsGraphicId& aOther) const
sl@0
   174
/** Compares another Id with this one.
sl@0
   175
	@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
sl@0
   176
	@publishedAll @released
sl@0
   177
*/	{
sl@0
   178
	if(iId < aOther.iId)
sl@0
   179
		{
sl@0
   180
		return -1;
sl@0
   181
		}
sl@0
   182
	else if(iId > aOther.iId)
sl@0
   183
		{
sl@0
   184
		return 1;
sl@0
   185
		}
sl@0
   186
	// else we have to compare the iIsUid flag too; again, expect it to be a match 99.99% of these times
sl@0
   187
	else if(IsUid() == aOther.IsUid())
sl@0
   188
		{
sl@0
   189
		return 0;
sl@0
   190
		}
sl@0
   191
	// collisions of id but not iIsUid are going to be really really rare
sl@0
   192
	else if(IsUid())
sl@0
   193
		{
sl@0
   194
		return 1;
sl@0
   195
		}
sl@0
   196
	else
sl@0
   197
		{
sl@0
   198
		return -1;
sl@0
   199
		}
sl@0
   200
	}
sl@0
   201
sl@0
   202
// CWsGraphicManager \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
   203
sl@0
   204
CWsGraphic::CManager* CWsGraphic::CManager::StaticL()
sl@0
   205
	{
sl@0
   206
	CManager* singleton = RWsBuffer::WsGraphicManager();
sl@0
   207
	if(!singleton)
sl@0
   208
		{
sl@0
   209
		singleton = new(ELeave) CManager;
sl@0
   210
		CleanupStack::PushL(singleton);
sl@0
   211
		singleton->ConstructL();
sl@0
   212
		CleanupStack::Pop(singleton);
sl@0
   213
		__ASSERT_DEBUG(singleton == RWsBuffer::WsGraphicManager(),Panic(EW32PanicGraphicInternal));
sl@0
   214
		}
sl@0
   215
	return singleton;
sl@0
   216
	}
sl@0
   217
sl@0
   218
CWsGraphic::CManager::~CManager()
sl@0
   219
	{
sl@0
   220
	__ASSERT_DEBUG(!ResourceCount(),Panic(EW32PanicGraphicInternal));
sl@0
   221
	__ASSERT_DEBUG(!iArray.Count(),Panic(EW32PanicGraphicOrphaned));
sl@0
   222
	Close();
sl@0
   223
	__ASSERT_DEBUG(!RWsBuffer::WsGraphicManager(),Panic(EW32PanicGraphicInternal));
sl@0
   224
	iArray.Close();
sl@0
   225
	delete iFlusher;
sl@0
   226
	}
sl@0
   227
sl@0
   228
CWsGraphic::CManager::CManager(): CActive(CActive::EPriorityStandard)
sl@0
   229
	{
sl@0
   230
	}
sl@0
   231
sl@0
   232
void CWsGraphic::CManager::ConstructL()
sl@0
   233
	{
sl@0
   234
	User::LeaveIfError(Connect());
sl@0
   235
	iBuffer->SetWsGraphicManager(this);
sl@0
   236
	CActiveScheduler::Add(this);
sl@0
   237
	iFlusher = CIdle::NewL(CActive::EPriorityIdle);
sl@0
   238
	}
sl@0
   239
sl@0
   240
void CWsGraphic::CManager::Inc()
sl@0
   241
	{
sl@0
   242
	iRefCount++;
sl@0
   243
	}
sl@0
   244
sl@0
   245
void CWsGraphic::CManager::Dec()
sl@0
   246
	{
sl@0
   247
	if(!--iRefCount)
sl@0
   248
		{
sl@0
   249
		delete this;
sl@0
   250
		}
sl@0
   251
	}
sl@0
   252
sl@0
   253
RWsBuffer* CWsGraphic::CManager::Buffer()
sl@0
   254
	{
sl@0
   255
	return iBuffer;
sl@0
   256
	}
sl@0
   257
sl@0
   258
// used by CWsGraphic
sl@0
   259
sl@0
   260
void CWsGraphic::CManager::AddL(CWsGraphic* aGraphic)
sl@0
   261
/** Leaves if the graphic couldn't be added to the list
sl@0
   262
	@internalComponent @released */
sl@0
   263
	{
sl@0
   264
	__ASSERT_ALWAYS(aGraphic && aGraphic->iPimpl,Panic(EW32PanicGraphicInternal));
sl@0
   265
	__ASSERT_ALWAYS(aGraphic->Id().IsId() || aGraphic->Id().IsUid(),Panic(EW32PanicGraphicInternal));
sl@0
   266
	__ASSERT_ALWAYS(!(aGraphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered),Panic(EW32PanicGraphicInternal));
sl@0
   267
	iArray.InsertInOrderL(aGraphic,GraphicCompare);
sl@0
   268
	__ASSERT_ALWAYS(0 <= iArray.FindInOrder(aGraphic,GraphicCompare),Panic(EW32PanicGraphicInternal));
sl@0
   269
	aGraphic->iPimpl->iFlags |= CWsGraphic::CPimpl::ERegistered;
sl@0
   270
	Queue();
sl@0
   271
	}
sl@0
   272
sl@0
   273
void CWsGraphic::CManager::Replace(CWsGraphic* aGraphic)
sl@0
   274
/** @internalComponent @released */
sl@0
   275
	{
sl@0
   276
	__ASSERT_ALWAYS(aGraphic && aGraphic->iPimpl,Panic(EW32PanicGraphicInternal));
sl@0
   277
	__ASSERT_ALWAYS(!(aGraphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered),Panic(EW32PanicGraphicInternal));
sl@0
   278
	__ASSERT_ALWAYS(aGraphic->Id().IsId() || aGraphic->Id().IsUid(),Panic(EW32PanicGraphicInternal));
sl@0
   279
	const TInt idx = iArray.FindInOrder(aGraphic,GraphicCompare);
sl@0
   280
	__ASSERT_ALWAYS(0 <= idx,Panic(EW32PanicGraphicInternal));
sl@0
   281
	__ASSERT_ALWAYS(iArray[idx]->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered,Panic(EW32PanicGraphicInternal));
sl@0
   282
	iArray[idx]->iPimpl->iFlags &= ~CWsGraphic::CPimpl::ERegistered;
sl@0
   283
	iArray[idx] = aGraphic;
sl@0
   284
	iArray[idx]->iPimpl->iFlags |= CWsGraphic::CPimpl::ERegistered;
sl@0
   285
	Queue();
sl@0
   286
	}
sl@0
   287
sl@0
   288
void CWsGraphic::CManager::Remove(CWsGraphic* aGraphic)
sl@0
   289
/** @internalComponent @released */
sl@0
   290
	{
sl@0
   291
	__ASSERT_ALWAYS(aGraphic && aGraphic->iPimpl,Panic(EW32PanicGraphicInternal));
sl@0
   292
	__ASSERT_ALWAYS(aGraphic->Id().IsId() || aGraphic->Id().IsUid(),Panic(EW32PanicGraphicInternal));
sl@0
   293
	__ASSERT_ALWAYS(aGraphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered,Panic(EW32PanicGraphicInternal));
sl@0
   294
	const TInt idx = iArray.FindInOrder(aGraphic,GraphicCompare);
sl@0
   295
	__ASSERT_ALWAYS(0 <= idx,Panic(EW32PanicGraphicInternal));
sl@0
   296
	iArray[idx]->iPimpl->iFlags &= ~CWsGraphic::CPimpl::ERegistered;
sl@0
   297
	iArray.Remove(idx);
sl@0
   298
	if(!iArray.Count())
sl@0
   299
		{
sl@0
   300
		Cancel();
sl@0
   301
		}
sl@0
   302
	}
sl@0
   303
sl@0
   304
CWsGraphic* CWsGraphic::CManager::Find(const TWsGraphicId& aId)
sl@0
   305
/** Find the active artwork identified by the Id
sl@0
   306
	@return NULL if no active artwork has the Id
sl@0
   307
	@publishedAll @released */
sl@0
   308
	{
sl@0
   309
	/*	RPointerArray can only FindInOrder other T*, which is a needless shame.  Therefore this search is
sl@0
   310
		dumb sequential */
sl@0
   311
	const TInt count = iArray.Count();
sl@0
   312
	for(TInt i=0; i<count; i++)
sl@0
   313
		{
sl@0
   314
		CWsGraphic* graphic = iArray[i];
sl@0
   315
		__ASSERT_ALWAYS(graphic && graphic->iPimpl && (graphic->iPimpl->iFlags & CWsGraphic::CPimpl::ERegistered),Panic(EW32PanicGraphicInternal));
sl@0
   316
		const TWsGraphicId& candidate = graphic->Id();
sl@0
   317
		if(0 == candidate.Compare(aId))
sl@0
   318
			{
sl@0
   319
			// found
sl@0
   320
			__ASSERT_DEBUG(i == iArray.FindInOrder(graphic,GraphicCompare),Panic(EW32PanicGraphicInternal));
sl@0
   321
			return graphic;
sl@0
   322
			}
sl@0
   323
		}
sl@0
   324
	// not found
sl@0
   325
	return NULL;
sl@0
   326
	}
sl@0
   327
sl@0
   328
void CWsGraphic::CManager::Queue()
sl@0
   329
	{
sl@0
   330
	if(!IsActive())
sl@0
   331
		{
sl@0
   332
		GraphicMessageReady(&iStatus);
sl@0
   333
		SetActive();
sl@0
   334
		}
sl@0
   335
	}
sl@0
   336
sl@0
   337
void CWsGraphic::CManager::RunL()
sl@0
   338
   	{
sl@0
   339
   	if(0 < iStatus.Int())
sl@0
   340
   		{
sl@0
   341
   		// a message to fetch!
sl@0
   342
   		const TInt msgLen = iStatus.Int();
sl@0
   343
 		TPtr8 buf(NULL,0);
sl@0
   344
 		MWsGraphicMessageAlloc* allocator = NULL;
sl@0
   345
 		MWsGraphicMessageAlloc::MBuffer* theirBuf = NULL;
sl@0
   346
 		HBufC8* ourBuf = HBufC8::New(msgLen);
sl@0
   347
 		if(ourBuf)
sl@0
   348
 			{
sl@0
   349
 			CleanupStack::PushL(ourBuf);
sl@0
   350
 			buf.Set(ourBuf->Des());
sl@0
   351
 			}
sl@0
   352
 		else // try to see if the destination CWsGraphic can allocate for us
sl@0
   353
 			{
sl@0
   354
			const TInt handle = GraphicFetchHeaderMessage();
sl@0
   355
			CWsGraphic* dest = reinterpret_cast<CWsGraphic*>(handle & ~0x03);
sl@0
   356
 			// check if it's valid
sl@0
   357
 			if(KErrNotFound != iArray.Find(dest) && dest->iPimpl && dest->iPimpl->iExt)
sl@0
   358
 				{
sl@0
   359
 				// check if the client is able to alloc memory for us
sl@0
   360
 				allocator = dest->iPimpl->iExt->ObjectInterface<MWsGraphicMessageAlloc>();
sl@0
   361
 				if(allocator)
sl@0
   362
 					{
sl@0
   363
 					// allocate memory
sl@0
   364
 					theirBuf = allocator->Alloc(msgLen);
sl@0
   365
 					if(theirBuf)
sl@0
   366
 						{
sl@0
   367
 						CleanupStack::PushL(TCleanupItem(MWsGraphicMessageAllocRelease,theirBuf));
sl@0
   368
 						buf.Set(theirBuf->Buffer());
sl@0
   369
 						}
sl@0
   370
 					}
sl@0
   371
 				}
sl@0
   372
 			}
sl@0
   373
 		if(!ourBuf && !theirBuf)
sl@0
   374
 			{
sl@0
   375
 			GraphicMessageCancel();
sl@0
   376
 			GraphicAbortMessage(KErrNoMemory);
sl@0
   377
 			}
sl@0
   378
 		else
sl@0
   379
 			{
sl@0
   380
 			GetGraphicMessage(buf);
sl@0
   381
 			// decode header and body
sl@0
   382
 			RDesReadStream in(buf);
sl@0
   383
 			in.PushL();
sl@0
   384
 			const TInt header = in.ReadInt32L();
sl@0
   385
 			__ASSERT_COMPILE(sizeof(header) == sizeof(TInt32));
sl@0
   386
 			const TInt clientHandle = (header & ~0x03);
sl@0
   387
 			const TInt msgType = (header & 0x03);
sl@0
   388
 			const TPtr8 body = buf.MidTPtr(sizeof(header));
sl@0
   389
 			// dispatch
sl@0
   390
 			CWsGraphic* dest = (CWsGraphic*)clientHandle;
sl@0
   391
 			if(KErrNotFound != iArray.Find(dest))
sl@0
   392
 				{
sl@0
   393
 				switch(msgType)
sl@0
   394
 					{
sl@0
   395
 					case EWsGraphMessageTypeUser:
sl@0
   396
 						dest->HandleMessage(body);
sl@0
   397
 						break;
sl@0
   398
 					default:
sl@0
   399
 						Panic(EW32PanicGraphicInternal);
sl@0
   400
 					}
sl@0
   401
 				}
sl@0
   402
 			// done
sl@0
   403
 			in.Pop();
sl@0
   404
 			}
sl@0
   405
 		if(ourBuf)
sl@0
   406
 			{
sl@0
   407
 			CleanupStack::PopAndDestroy(ourBuf);
sl@0
   408
 			}
sl@0
   409
 		else if(theirBuf)
sl@0
   410
 			{
sl@0
   411
 			CleanupStack::PopAndDestroy(theirBuf);
sl@0
   412
 			}
sl@0
   413
 		// done, wait for next message
sl@0
   414
 		Queue();
sl@0
   415
 		}
sl@0
   416
 	}
sl@0
   417
sl@0
   418
void CWsGraphic::CManager::DoCancel()
sl@0
   419
	{
sl@0
   420
	GraphicMessageCancel();
sl@0
   421
	}
sl@0
   422
sl@0
   423
TInt CWsGraphic::CManager::GraphicCompare(const CWsGraphic& aFirst,const CWsGraphic& aSecond)
sl@0
   424
/** Compares two graphics for id equality
sl@0
   425
@internalComponent
sl@0
   426
@released
sl@0
   427
*/	{
sl@0
   428
	return aFirst.Id().Compare(aSecond.Id());
sl@0
   429
	}
sl@0
   430
sl@0
   431
void CWsGraphic::CManager::ScheduleFlush()
sl@0
   432
/** Request to schedule flush when idle
sl@0
   433
@internalComponent
sl@0
   434
@released
sl@0
   435
*/
sl@0
   436
	{
sl@0
   437
	if (iFlushScheduled)
sl@0
   438
		return;
sl@0
   439
sl@0
   440
	iFlushScheduled = ETrue;
sl@0
   441
	iFlusher->Start(TCallBack(CWsGraphic::CManager::FlushOnIdle,this));
sl@0
   442
	}
sl@0
   443
sl@0
   444
TInt CWsGraphic::CManager::FlushOnIdle(TAny* aArg)
sl@0
   445
/** Flush buffer when idle and there is outstanding data in it
sl@0
   446
@internalComponent
sl@0
   447
@released
sl@0
   448
*/
sl@0
   449
	{
sl@0
   450
	CWsGraphic::CManager* mgr = reinterpret_cast<CWsGraphic::CManager*>(aArg);
sl@0
   451
	if (mgr)
sl@0
   452
		{
sl@0
   453
		mgr->iFlushScheduled = EFalse;
sl@0
   454
		if (mgr->iBuffer && !mgr->iBuffer->IsEmpty())
sl@0
   455
			mgr->iBuffer->Flush();
sl@0
   456
		}
sl@0
   457
sl@0
   458
	return 0; // complete
sl@0
   459
	}
sl@0
   460
sl@0
   461
// CWsGraphic::CPimpl \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
   462
sl@0
   463
CWsGraphic::CPimpl::CPimpl(CWsGraphic& aGraphic):
sl@0
   464
	iGraphic(aGraphic), iId(TWsGraphicId::EUninitialized)
sl@0
   465
	{
sl@0
   466
	}
sl@0
   467
sl@0
   468
CWsGraphic::CPimpl::~CPimpl()
sl@0
   469
	{
sl@0
   470
	if(iManager)
sl@0
   471
		{
sl@0
   472
		iManager->Dec();
sl@0
   473
		}
sl@0
   474
	}
sl@0
   475
sl@0
   476
void CWsGraphic::CPimpl::ConstructL()
sl@0
   477
	{
sl@0
   478
	iManager = CManager::StaticL();
sl@0
   479
	iManager->Inc();
sl@0
   480
	iBuffer = iManager->Buffer();
sl@0
   481
	}
sl@0
   482
sl@0
   483
TInt CWsGraphic::CPimpl::WriteCreateGraphic(TWsClCmdCreateGraphic& aCreateGraphic,const TDesC8& aData,TInt aWsHandle) const
sl@0
   484
/** Writes the CreateGraphic message to the server.  If the data will not fit in the buffer, uses remote-read
sl@0
   485
	@internalComponent @released */
sl@0
   486
	{
sl@0
   487
	aCreateGraphic.iDataLen = aData.Size();
sl@0
   488
	aCreateGraphic.iRemoteReadData = ((aData.Size()+sizeof(aCreateGraphic))>(TInt)(iBuffer->BufferSize()-sizeof(TWsCmdHeader)));
sl@0
   489
	if(aCreateGraphic.iRemoteReadData)
sl@0
   490
		{
sl@0
   491
		return iBuffer->WriteReplyByProvidingRemoteReadAccess(aWsHandle,EWsClOpCreateGraphic,&aCreateGraphic,sizeof(aCreateGraphic),&aData);
sl@0
   492
		}
sl@0
   493
	else if(aCreateGraphic.iDataLen)
sl@0
   494
		{
sl@0
   495
		return iBuffer->WriteReplyWs(&aCreateGraphic,sizeof(aCreateGraphic),aData.Ptr(),aData.Size(),EWsClOpCreateGraphic);
sl@0
   496
		}
sl@0
   497
	else
sl@0
   498
		{
sl@0
   499
		return iBuffer->WriteReplyWs(&aCreateGraphic,sizeof(aCreateGraphic),EWsClOpCreateGraphic);
sl@0
   500
		}
sl@0
   501
	}
sl@0
   502
sl@0
   503
// CWsGraphic \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
   504
sl@0
   505
/** 
sl@0
   506
Default Constructor.
sl@0
   507
*/
sl@0
   508
EXPORT_C CWsGraphic::CWsGraphic()
sl@0
   509
	{
sl@0
   510
	}
sl@0
   511
sl@0
   512
/**
sl@0
   513
Destructor.
sl@0
   514
*/
sl@0
   515
EXPORT_C CWsGraphic::~CWsGraphic()
sl@0
   516
	{
sl@0
   517
	Destroy();
sl@0
   518
	delete iPimpl;
sl@0
   519
	}
sl@0
   520
	
sl@0
   521
/**
sl@0
   522
Completes construction of the baseclass. All the overloaded BaseConstructL() methods
sl@0
   523
should invoke this method to complete the construction of the baseclass.
sl@0
   524
*/
sl@0
   525
void CWsGraphic::BaseConstructL()
sl@0
   526
	{
sl@0
   527
	iPimpl = new(ELeave) CPimpl(*this);
sl@0
   528
	iPimpl->ConstructL();
sl@0
   529
	}
sl@0
   530
sl@0
   531
EXPORT_C void CWsGraphic::BaseConstructL(TUid aUid,TUid aType,const TDesC8& aData)
sl@0
   532
/** 
sl@0
   533
Constructs a piece of non-transient graphic artwork.
sl@0
   534
@capability ProtServ
sl@0
   535
@param aUid	Graphic artwork UID.
sl@0
   536
@param aType Graphic artwork type.
sl@0
   537
@param aData User specific data.
sl@0
   538
*/	{
sl@0
   539
	BaseConstructL();
sl@0
   540
sl@0
   541
	TWsClCmdCreateGraphic createGraphic;
sl@0
   542
	createGraphic.iFlags = EWsGraphicIdUid;
sl@0
   543
	createGraphic.iId = aUid.iUid;
sl@0
   544
	createGraphic.iType = aType;
sl@0
   545
	createGraphic.iClientHandle = (TInt)this;
sl@0
   546
sl@0
   547
	TInt ret = iPimpl->WriteCreateGraphic(createGraphic,aData,iPimpl->iManager->WsHandle());
sl@0
   548
	User::LeaveIfError(ret);
sl@0
   549
sl@0
   550
	iPimpl->iWsHandle=ret;
sl@0
   551
	iPimpl->iId = aUid;
sl@0
   552
	iPimpl->iFlags = CPimpl::EHasPeer;
sl@0
   553
	iPimpl->iManager->AddL(this);
sl@0
   554
	}
sl@0
   555
sl@0
   556
EXPORT_C void CWsGraphic::BaseConstructL(TUid aType,const TDesC8& aData)
sl@0
   557
/** 
sl@0
   558
Constructs a piece of transient graphic artwork.
sl@0
   559
@param aType Graphic artwork type.
sl@0
   560
@param aData User specific data.
sl@0
   561
*/
sl@0
   562
	{
sl@0
   563
	BaseConstructL();
sl@0
   564
sl@0
   565
	TWsClCmdCreateGraphic createGraphic;
sl@0
   566
	createGraphic.iFlags = EWsGraphicIdTransient;
sl@0
   567
	createGraphic.iId = 0;
sl@0
   568
	createGraphic.iType = aType;
sl@0
   569
	createGraphic.iClientHandle = (TInt)this;
sl@0
   570
sl@0
   571
	TInt ret = iPimpl->WriteCreateGraphic(createGraphic,aData,iPimpl->iManager->WsHandle());
sl@0
   572
	User::LeaveIfError(ret);
sl@0
   573
sl@0
   574
	iPimpl->iWsHandle = ret;
sl@0
   575
sl@0
   576
	// fetch id from server
sl@0
   577
	TPckgBuf<TWsClCmdGdGetId> cmd;
sl@0
   578
	User::LeaveIfError(iPimpl->WriteReplyP(TWriteDescriptorType(&cmd),EWsGdOpGetGraphicId));
sl@0
   579
	if(cmd().iIsUid)
sl@0
   580
		{
sl@0
   581
		__DEBUG_ONLY(Panic(EW32PanicGraphicInternal));
sl@0
   582
		User::Leave(KErrGeneral);
sl@0
   583
		}
sl@0
   584
	iPimpl->iId = cmd().iId;
sl@0
   585
sl@0
   586
	iPimpl->iFlags = CPimpl::EHasPeer;
sl@0
   587
	iPimpl->iManager->AddL(this);
sl@0
   588
	}
sl@0
   589
sl@0
   590
EXPORT_C void CWsGraphic::BaseConstructL(const TWsGraphicId& aReplace,TUid aType,const TDesC8& aData)
sl@0
   591
/** 
sl@0
   592
Atomically replace the artwork that already exists with this artwork.
sl@0
   593
If failure to properly construct the replacement artwork occurs, the replacee artwork will remain
sl@0
   594
@param aReplace Graphic artwork which will be replaced.
sl@0
   595
@param aType New graphic artwork type.
sl@0
   596
@param aData User specific data.
sl@0
   597
*/
sl@0
   598
	{
sl@0
   599
	BaseConstructL();
sl@0
   600
sl@0
   601
	CWsGraphic* dup = iPimpl->iManager->Find(aReplace);
sl@0
   602
	if(!dup || !dup->iPimpl)
sl@0
   603
		{
sl@0
   604
		Panic(EW32PanicGraphicInternal);
sl@0
   605
		}
sl@0
   606
sl@0
   607
	const TUint flags = aReplace.IsUid()?EWsGraphicIdUid:EWsGraphicIdTransient;
sl@0
   608
sl@0
   609
	TWsClCmdCreateGraphic createGraphic;
sl@0
   610
	createGraphic.iFlags = EWsGraphicReplace|flags;
sl@0
   611
	createGraphic.iId = aReplace.IsUid()?aReplace.Uid().iUid:aReplace.Id();
sl@0
   612
	createGraphic.iType = aType;
sl@0
   613
	createGraphic.iClientHandle = (TInt)this;
sl@0
   614
sl@0
   615
	TInt ret = iPimpl->WriteCreateGraphic(createGraphic,aData,iPimpl->iManager->WsHandle());
sl@0
   616
	if(0 > ret)
sl@0
   617
		{
sl@0
   618
		User::Leave(ret);
sl@0
   619
		}
sl@0
   620
sl@0
   621
	iPimpl->iWsHandle = ret;
sl@0
   622
	iPimpl->iId = aReplace;
sl@0
   623
sl@0
   624
	iPimpl->iFlags = CPimpl::EHasPeer;
sl@0
   625
	iPimpl->iManager->Replace(this);
sl@0
   626
sl@0
   627
	// when WriteCreateGraphic succeeds, the replacee drawer has already been destroyed serverside
sl@0
   628
	// so this cleanup is not quite the same as Destroy(), as it doesn't free server side
sl@0
   629
	dup->iPimpl->iWsHandle = 0;
sl@0
   630
	dup->OnReplace();
sl@0
   631
	dup->iPimpl->iFlags &= ~CPimpl::EHasPeer;
sl@0
   632
	}
sl@0
   633
sl@0
   634
/**
sl@0
   635
Shares the graphic artwork with all the client sessions.
sl@0
   636
Sharing globally trumps explicit shares.
sl@0
   637
@return KErrNone if the graphic is globally shared, else one of the system-wide error codes.
sl@0
   638
*/
sl@0
   639
EXPORT_C TInt CWsGraphic::ShareGlobally()
sl@0
   640
	{
sl@0
   641
	if(!IsActive())
sl@0
   642
		{
sl@0
   643
		return KErrNotReady;
sl@0
   644
		}
sl@0
   645
	return iPimpl->WriteReply(EWsGdOpShareGlobally);
sl@0
   646
	}
sl@0
   647
sl@0
   648
/**
sl@0
   649
Prevents this graphic artwork from being shared with all the client sessions.
sl@0
   650
A graphic artwork that isn't shared explicitly is only available to clients it
sl@0
   651
has been explicitly shared with using Share().
sl@0
   652
@return KErrNone if the graphic is not globally shared, else one of the system-wide error codes.
sl@0
   653
*/
sl@0
   654
EXPORT_C TInt CWsGraphic::UnShareGlobally()
sl@0
   655
	{
sl@0
   656
	if(!IsActive())
sl@0
   657
		{
sl@0
   658
		return KErrNotReady;
sl@0
   659
		}
sl@0
   660
	return iPimpl->WriteReply(EWsGdOpUnShareGlobally);
sl@0
   661
	}
sl@0
   662
	
sl@0
   663
/** 
sl@0
   664
Explicitly shares this graphic artwork with client sessions with the specified Secure ID.
sl@0
   665
@param aClientId the Secure ID of the client sessions to share with.
sl@0
   666
@return KErrNone If the graphic artwork was shared, else one of the system-wide error codes.
sl@0
   667
*/
sl@0
   668
EXPORT_C TInt CWsGraphic::Share(TSecureId aClientId)
sl@0
   669
	{
sl@0
   670
	if(!IsActive())
sl@0
   671
		{
sl@0
   672
		return KErrNotReady;
sl@0
   673
		}
sl@0
   674
	return iPimpl->WriteReply(&aClientId,sizeof(TSecureId),EWsGdOpShare);
sl@0
   675
	}
sl@0
   676
sl@0
   677
/** 
sl@0
   678
Stops this graphic artwork from being shared with all client sessions with the specific Secure ID.
sl@0
   679
ShareGlobally() trumps explicit sharing.
sl@0
   680
@param aClientId the Secure ID of the client sessions to not share with
sl@0
   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
sl@0
   682
*/
sl@0
   683
EXPORT_C TInt CWsGraphic::UnShare(TSecureId aClientId)
sl@0
   684
	{
sl@0
   685
	if(!IsActive())
sl@0
   686
		{
sl@0
   687
		return KErrNotReady;
sl@0
   688
		}
sl@0
   689
	return iPimpl->WriteReply(&aClientId,sizeof(TSecureId),EWsGdOpUnShare);
sl@0
   690
	}
sl@0
   691
sl@0
   692
/**
sl@0
   693
Returns graphic artwork Id.
sl@0
   694
@return Graphic artwork Id. KNullWsGraphicId if graphic artwork is not active.
sl@0
   695
*/
sl@0
   696
EXPORT_C const TWsGraphicId& CWsGraphic::Id() const
sl@0
   697
	{
sl@0
   698
	if(IsActive())
sl@0
   699
		{
sl@0
   700
		return iPimpl->iId;
sl@0
   701
		}
sl@0
   702
	else
sl@0
   703
		{
sl@0
   704
		// fallback
sl@0
   705
		static const TInt KNullWsGraphicId[4] = //binary compatible with TWsGraphicId
sl@0
   706
			{
sl@0
   707
			0, 0, 0, 0
sl@0
   708
			};
sl@0
   709
		__ASSERT_COMPILE(sizeof(KNullWsGraphicId) == sizeof(TWsGraphicId));
sl@0
   710
		return reinterpret_cast<const TWsGraphicId&>(KNullWsGraphicId);
sl@0
   711
		}
sl@0
   712
	}
sl@0
   713
	
sl@0
   714
/** 
sl@0
   715
Checks whether a peer of this graphic artwork has been fully constructed on the server.
sl@0
   716
@return ETrue if this graphic artwork has a peer CWsGraphic on the server.
sl@0
   717
*/
sl@0
   718
EXPORT_C TBool CWsGraphic::IsActive() const
sl@0
   719
	{
sl@0
   720
	return (iPimpl && iPimpl->iWsHandle && (iPimpl->iFlags & CPimpl::EHasPeer));
sl@0
   721
	}
sl@0
   722
	
sl@0
   723
/**
sl@0
   724
Derived class can override this method to provide custom operations when the client is closed.
sl@0
   725
*/
sl@0
   726
EXPORT_C void CWsGraphic::OnClientClose()
sl@0
   727
	{
sl@0
   728
	}
sl@0
   729
	
sl@0
   730
/** 
sl@0
   731
Sends message to this graphic artwork peer on the server.
sl@0
   732
@param aData User specific data.
sl@0
   733
*/
sl@0
   734
EXPORT_C void CWsGraphic::SendMessage(const TDesC8& aData) const
sl@0
   735
	{
sl@0
   736
	TWsClCmdGdSendMessage cmd;
sl@0
   737
	cmd.iDataLen = aData.Size();
sl@0
   738
	__ASSERT_DEBUG(cmd.iDataLen, Panic(EW32PanicGraphicNullData));
sl@0
   739
	cmd.iRemoteReadData = ((aData.Size()+sizeof(cmd))>(TInt)(iPimpl->iBuffer->BufferSize()-sizeof(TWsCmdHeader)));
sl@0
   740
	if(cmd.iRemoteReadData)
sl@0
   741
		{
sl@0
   742
		iPimpl->WriteReplyByProvidingRemoteReadAccess(&cmd,sizeof(cmd),&aData,EWsGdOpSendMsg);
sl@0
   743
		//ignore return value!
sl@0
   744
		}
sl@0
   745
	else
sl@0
   746
		{
sl@0
   747
		iPimpl->Write(&cmd,sizeof(cmd),aData.Ptr(),aData.Size(),EWsGdOpSendMsg);
sl@0
   748
		}
sl@0
   749
	}
sl@0
   750
sl@0
   751
EXPORT_C TInt CWsGraphic::SendSynchronMessage(const TDesC8& aData) const
sl@0
   752
	 {
sl@0
   753
	 TWsClCmdGdSendMessage cmd;
sl@0
   754
	 cmd.iDataLen = aData.Size();
sl@0
   755
	 __ASSERT_DEBUG(cmd.iDataLen, Panic(EW32PanicGraphicNullData));
sl@0
   756
	 cmd.iRemoteReadData = ((aData.Size()+sizeof(cmd))>(TInt)(iPimpl->iBuffer->BufferSize()-sizeof(TWsCmdHeader)));
sl@0
   757
	 if(cmd.iRemoteReadData)
sl@0
   758
		{
sl@0
   759
		return iPimpl->WriteReplyByProvidingRemoteReadAccess(&cmd,sizeof(cmd),&aData,EWsGdOpSendSynchronMsg);
sl@0
   760
		}
sl@0
   761
	 else
sl@0
   762
		{
sl@0
   763
		return iPimpl->WriteReply(&cmd,sizeof(cmd),aData.Ptr(),aData.Size(),EWsGdOpSendSynchronMsg);
sl@0
   764
		}
sl@0
   765
	 }
sl@0
   766
sl@0
   767
/** 
sl@0
   768
Flushes window server command buffer
sl@0
   769
@return One of system-wide error codes.
sl@0
   770
*/
sl@0
   771
EXPORT_C TInt CWsGraphic::Flush() const
sl@0
   772
	{
sl@0
   773
	return iPimpl->iBuffer->Flush();
sl@0
   774
	}
sl@0
   775
sl@0
   776
EXPORT_C void CWsGraphic::Destroy()
sl@0
   777
/** Destroys the corresponding CWsGraphicDrawer instance on the server
sl@0
   778
@released
sl@0
   779
@publishedAll
sl@0
   780
*/	{
sl@0
   781
	if(iPimpl && (iPimpl->iFlags & CPimpl::ERegistered))
sl@0
   782
		{
sl@0
   783
		iPimpl->iManager->Remove(this);
sl@0
   784
		}
sl@0
   785
	if(IsActive()) // if iPimpl==NULL, IsActive() returns false
sl@0
   786
		{
sl@0
   787
		iPimpl->Write(EWsGdOpFree);
sl@0
   788
		iPimpl->iWsHandle = 0;
sl@0
   789
		iPimpl->iFlags &= ~CPimpl::EHasPeer;
sl@0
   790
		iPimpl->iManager->ScheduleFlush();
sl@0
   791
		}
sl@0
   792
	}
sl@0
   793
sl@0
   794
EXPORT_C void CWsGraphic::SetGraphicExtension(MWsObjectProvider* aExt)
sl@0
   795
	{
sl@0
   796
	iPimpl->iExt = aExt;
sl@0
   797
	}
sl@0
   798
sl@0
   799
EXPORT_C RWsSession&  CWsGraphic::Session()
sl@0
   800
	{
sl@0
   801
	return *iPimpl->iManager;
sl@0
   802
	}
sl@0
   803
sl@0
   804
EXPORT_C TInt CWsGraphic::CWsGraphic_Reserved1()
sl@0
   805
	{
sl@0
   806
	return KErrNotSupported;
sl@0
   807
	}
sl@0
   808
sl@0
   809
EXPORT_C TInt CWsGraphic::CWsGraphic_Reserved2()
sl@0
   810
	{
sl@0
   811
	return KErrNotSupported;
sl@0
   812
	}
sl@0
   813
sl@0
   814
EXPORT_C TInt CWsGraphic::CWsGraphic_Reserved3()
sl@0
   815
	{
sl@0
   816
	return KErrNotSupported;
sl@0
   817
	}
sl@0
   818
sl@0
   819
// TWsGraphicMsgFixedBase \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
   820
sl@0
   821
EXPORT_C TWsGraphicMsgFixedBase::TWsGraphicMsgFixedBase(TUid aTypeId,TInt aSizeOfDerived):
sl@0
   822
/** Protected constructor for subclasses to call
sl@0
   823
@param aTypeId The UID representing this type of data
sl@0
   824
@param aSizeOf The size of the derived class
sl@0
   825
sl@0
   826
Example:
sl@0
   827
@code
sl@0
   828
TMyDerivedFixedMsg::TMyDerivedFixedMsg(): TWsGraphicMsgFixedBase(KUidMyDerivedType,sizeof(TMyDerivedFixedMsg)), ...
sl@0
   829
@endcode
sl@0
   830
*/	iTypeId(aTypeId), iSize(aSizeOfDerived-sizeof(TWsGraphicMsgFixedBase))
sl@0
   831
	{
sl@0
   832
	__ASSERT_COMPILE(sizeof(*this) == (sizeof(TInt32)*2));
sl@0
   833
	}
sl@0
   834
sl@0
   835
EXPORT_C TPtrC8 TWsGraphicMsgFixedBase::Pckg() const
sl@0
   836
/** @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
sl@0
   837
*/	{
sl@0
   838
	return TPtrC8(reinterpret_cast<const TUint8*>(this),sizeof(*this) + iSize);
sl@0
   839
	}
sl@0
   840
sl@0
   841
EXPORT_C TUid TWsGraphicMsgFixedBase::TypeId() const
sl@0
   842
/** @return the UID identifying the type of the data that follows */
sl@0
   843
	{
sl@0
   844
	return iTypeId;
sl@0
   845
	}
sl@0
   846
sl@0
   847
EXPORT_C TInt TWsGraphicMsgFixedBase::Size() const
sl@0
   848
/** @return the size of the derived class (not including this fixed base class size) */
sl@0
   849
	{
sl@0
   850
	return iSize;
sl@0
   851
	}
sl@0
   852
sl@0
   853
// RWsGraphicMsgBuf \\\\\\\\\\\\\\\\\\\\\\\\
sl@0
   854
sl@0
   855
EXPORT_C RWsGraphicMsgBuf::RWsGraphicMsgBuf()
sl@0
   856
/** Default Constructor */
sl@0
   857
	{
sl@0
   858
	}
sl@0
   859
sl@0
   860
TInt RWsGraphicMsgBuf::IntAt(TInt aOfs) const
sl@0
   861
/** @internalComponent @released */
sl@0
   862
	{
sl@0
   863
	if((aOfs < 0) || ((aOfs+sizeof(TInt)) > Length()))
sl@0
   864
		{
sl@0
   865
		Panic(EW32PanicGraphicBadBuffer);
sl@0
   866
		}
sl@0
   867
	TInt ret;
sl@0
   868
	memcpy(&ret,Ptr()+aOfs,sizeof(TInt));
sl@0
   869
	return ret;
sl@0
   870
	}
sl@0
   871
sl@0
   872
EXPORT_C TInt RWsGraphicMsgBuf::Append(TUid aTypeId,const TDesC8& aData)
sl@0
   873
/** Append a descriptor as data
sl@0
   874
	@param aTypeId the type of the message to append
sl@0
   875
	@param aData arbitrary length data consisting of the whole message
sl@0
   876
	@return KErrNone if successful, else a system-wide error code
sl@0
   877
*/	{
sl@0
   878
	TInt err = ExpandForAppend(aData.Length());
sl@0
   879
	if (err)
sl@0
   880
		{
sl@0
   881
		return err;
sl@0
   882
		}
sl@0
   883
	WriteHeader(aTypeId,aData.Length());
sl@0
   884
	// append data
sl@0
   885
	Insert(Length(),aData);
sl@0
   886
	return KErrNone;
sl@0
   887
	}
sl@0
   888
sl@0
   889
EXPORT_C TInt RWsGraphicMsgBuf::Append(TUid aTypeId,const TDesC16& aData)
sl@0
   890
/** Append a descriptor as data
sl@0
   891
	@param aTypeId the type of the message to append
sl@0
   892
	@param aData arbitrary length data consisting of the whole message
sl@0
   893
	@return KErrNone if successful, else a system-wide error code
sl@0
   894
*/	{
sl@0
   895
	TPtr8 data(NULL,0);
sl@0
   896
	TInt err = Append(aTypeId,aData.Size(),data);
sl@0
   897
	if (err)
sl@0
   898
		{
sl@0
   899
		return err;
sl@0
   900
		}
sl@0
   901
	// append data
sl@0
   902
	data.Copy(reinterpret_cast<const TUint8*>(aData.Ptr()),aData.Size());
sl@0
   903
	return KErrNone;
sl@0
   904
	}
sl@0
   905
sl@0
   906
TInt RWsGraphicMsgBuf::ExpandForAppend(TInt aDataLen)
sl@0
   907
/** @internalComponent @released */
sl@0
   908
	{
sl@0
   909
	__ASSERT_COMPILE(sizeof(TInt) == sizeof(TInt32));
sl@0
   910
	const TInt required = (sizeof(TUid) + sizeof(TInt) + aDataLen);
sl@0
   911
	if(MaxLength() < (Length() + required))
sl@0
   912
		{
sl@0
   913
		TInt err = ReAlloc(Length()+required);
sl@0
   914
		if(KErrNone != err)
sl@0
   915
			{
sl@0
   916
			return err;
sl@0
   917
			}
sl@0
   918
		}
sl@0
   919
	return KErrNone;
sl@0
   920
	}
sl@0
   921
sl@0
   922
void RWsGraphicMsgBuf::WriteHeader(TUid aTypeId,TInt aLen)
sl@0
   923
/** @internalComponent @released */
sl@0
   924
	{
sl@0
   925
	__ASSERT_COMPILE(sizeof(TInt) == sizeof(TInt32));
sl@0
   926
	// append header
sl@0
   927
	TPckgBuf<TInt32> i(aTypeId.iUid);
sl@0
   928
	Insert(Length(),i);
sl@0
   929
	i() = aLen;
sl@0
   930
	Insert(Length(),i);
sl@0
   931
	}
sl@0
   932
sl@0
   933
EXPORT_C TInt RWsGraphicMsgBuf::Append(TUid aTypeId,TInt aLen,TPtr8& aPtr)
sl@0
   934
/** Append a message of the specified length and type, and return a pointer to
sl@0
   935
	allow client code to modify the message.
sl@0
   936
	aPtr is only set if the append succeeds.
sl@0
   937
	aPtr is only valid until the next message is appended to the buffer.
sl@0
   938
	@param aTypeId the type of the message to append
sl@0
   939
	@param aLen the length of the message to be reserved
sl@0
   940
	@param aPtr a modifiable descriptor to be used by the client code to write into the message body
sl@0
   941
	@return KErrNone if successful, else a system-wide error code
sl@0
   942
*/	{
sl@0
   943
	TInt err = ExpandForAppend(aLen);
sl@0
   944
	if (err)
sl@0
   945
		{
sl@0
   946
		return err;
sl@0
   947
		}
sl@0
   948
	WriteHeader(aTypeId,aLen);
sl@0
   949
	// set aPtr
sl@0
   950
	TInt usedLen = Length();
sl@0
   951
	SetLength(usedLen+aLen);
sl@0
   952
	aPtr.Set(MidTPtr(usedLen,aLen));
sl@0
   953
	aPtr.Fill('_'); // prettier when debugging, but want consistant behaviour in release builds
sl@0
   954
	aPtr.Zero();
sl@0
   955
	return KErrNone;
sl@0
   956
	}
sl@0
   957
sl@0
   958
EXPORT_C void RWsGraphicMsgBuf::Remove(TInt aIndex)
sl@0
   959
/** Remove a message from the buffer
sl@0
   960
	@panic if the index is out of bounds
sl@0
   961
	@param aIndex the ordinal position of message to be removed
sl@0
   962
*/	{
sl@0
   963
	if((aIndex < 0) || (aIndex >= Count()))
sl@0
   964
		{
sl@0
   965
		Panic(EW32PanicGraphicBadBuffer);
sl@0
   966
		}
sl@0
   967
	TPtrC8 ptr = Data(aIndex);
sl@0
   968
	const TInt ofs = (ptr.Ptr()-(sizeof(TInt)*2)-Ptr());
sl@0
   969
	Delete(ofs,ptr.Length() + (sizeof(TInt)*2));
sl@0
   970
	}
sl@0
   971
sl@0
   972
EXPORT_C TInt RWsGraphicMsgBuf::Count() const
sl@0
   973
/** Returns the number of messages in the buffer
sl@0
   974
	@return the number of messages in the buffer
sl@0
   975
*/	{
sl@0
   976
	const TInt length = Length();
sl@0
   977
	TInt ofs = 0, count = 0;
sl@0
   978
	while(ofs < length)
sl@0
   979
		{
sl@0
   980
		count++;
sl@0
   981
		ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
sl@0
   982
		}
sl@0
   983
	if(ofs != length)
sl@0
   984
		{
sl@0
   985
		Panic(EW32PanicGraphicBadBuffer);
sl@0
   986
		}
sl@0
   987
	return count;
sl@0
   988
	}
sl@0
   989
sl@0
   990
EXPORT_C TUid RWsGraphicMsgBuf::TypeId(TInt aIndex) const
sl@0
   991
/** Returns the Type ID of a message in the buffer
sl@0
   992
	@param aIndex the ordinal position of the message
sl@0
   993
	@return the Type ID of the message
sl@0
   994
	@panic if the index is out of bounds
sl@0
   995
*/	{
sl@0
   996
	const TInt length = Length();
sl@0
   997
	TInt ofs = 0, count = 0;
sl@0
   998
	while(ofs < length)
sl@0
   999
		{
sl@0
  1000
		if(count == aIndex)
sl@0
  1001
			{
sl@0
  1002
			return TUid::Uid(IntAt(ofs));
sl@0
  1003
			}
sl@0
  1004
		count++;
sl@0
  1005
		ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
sl@0
  1006
		}
sl@0
  1007
	Panic(EW32PanicGraphicBadBuffer);
sl@0
  1008
	return KNullUid; //dumb compiler
sl@0
  1009
	}
sl@0
  1010
sl@0
  1011
EXPORT_C TPtrC8 RWsGraphicMsgBuf::Data(TInt aIndex) const
sl@0
  1012
/** Returns a non-modifiable descriptor of a message body in the buffer
sl@0
  1013
	@param aIndex the ordinal position of the message
sl@0
  1014
	@return the message body
sl@0
  1015
	@panic if the index is out of bounds
sl@0
  1016
*/	{
sl@0
  1017
	const TInt length = Length();
sl@0
  1018
	TInt ofs = 0, count = 0;
sl@0
  1019
	while(ofs < length)
sl@0
  1020
		{
sl@0
  1021
		if(count == aIndex)
sl@0
  1022
			{
sl@0
  1023
			return Mid(ofs+(sizeof(TInt)*2),IntAt(ofs+sizeof(TInt)));
sl@0
  1024
			}
sl@0
  1025
		count++;
sl@0
  1026
		ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
sl@0
  1027
		}
sl@0
  1028
	Panic(EW32PanicGraphicBadBuffer);
sl@0
  1029
	return TPtrC8(KNullDesC8()); //dumb compiler
sl@0
  1030
	}
sl@0
  1031
sl@0
  1032
EXPORT_C TPtr8 RWsGraphicMsgBuf::Data(TInt aIndex)
sl@0
  1033
/** Returns a modifiable descriptor of a message body in the buffer
sl@0
  1034
	The returned TPtr8 is only valid until the next message is appended to the buffer.
sl@0
  1035
	@param aIndex the ordinal position of the message
sl@0
  1036
	@return the message body
sl@0
  1037
	@panic if the index is out of bounds
sl@0
  1038
*/	{
sl@0
  1039
	const TInt length = Length();
sl@0
  1040
	TInt ofs = 0, count = 0;
sl@0
  1041
	while(ofs < length)
sl@0
  1042
		{
sl@0
  1043
		if(count == aIndex)
sl@0
  1044
			{
sl@0
  1045
			return MidTPtr(ofs+(sizeof(TInt)*2),IntAt(ofs+sizeof(TInt)));
sl@0
  1046
			}
sl@0
  1047
		count++;
sl@0
  1048
		ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
sl@0
  1049
		}
sl@0
  1050
	Panic(EW32PanicGraphicBadBuffer);
sl@0
  1051
	return TPtr8(NULL,0); //dumb compiler
sl@0
  1052
	}
sl@0
  1053
sl@0
  1054
EXPORT_C const TDesC8& RWsGraphicMsgBuf::Pckg() const
sl@0
  1055
/** Returns the message buffer as a descriptor.  Example:
sl@0
  1056
	@code
sl@0
  1057
	RWsGraphicMsgBuf msgBuf;
sl@0
  1058
	msgBuf.Append(...);
sl@0
  1059
	...
sl@0
  1060
	TWsGraphicId id(...);
sl@0
  1061
	SystemGc().DrawWsGraphic(id,Rect(),msgBuf.Pckg());
sl@0
  1062
	msgBuf.Close();
sl@0
  1063
	@endcode
sl@0
  1064
	@see CWindowGc::DrawWsGraphic
sl@0
  1065
	@see CCoeControl::Draw
sl@0
  1066
	@return the message buffer to be attached a command to draw a CWsGraphic
sl@0
  1067
*/	{
sl@0
  1068
	return *this;
sl@0
  1069
	}
sl@0
  1070
sl@0
  1071
EXPORT_C TInt RWsGraphicMsgBuf::Append(const TWsGraphicMsgFixedBase& aMsg)
sl@0
  1072
/** Append a fixed-size message
sl@0
  1073
	@param aMsg the fixed-size message to append
sl@0
  1074
	@return KErrNone if successful, else a system-wide error code
sl@0
  1075
*/	{
sl@0
  1076
	__ASSERT_COMPILE(sizeof(TWsGraphicMsgFixedBase) == (sizeof(TInt)*2));
sl@0
  1077
	TInt err = ExpandForAppend(aMsg.Size());
sl@0
  1078
	if (err)
sl@0
  1079
		{
sl@0
  1080
		return err;
sl@0
  1081
		}
sl@0
  1082
	// append data
sl@0
  1083
	RBuf8::Append(reinterpret_cast<const TUint8*>(&aMsg),sizeof(TWsGraphicMsgFixedBase) + aMsg.Size());
sl@0
  1084
	Count();
sl@0
  1085
	return KErrNone;
sl@0
  1086
	}
sl@0
  1087
sl@0
  1088
EXPORT_C void RWsGraphicMsgBuf::GetFixedMsg(TWsGraphicMsgFixedBase& aMsg,TInt aIndex) const
sl@0
  1089
/** Returns a copy of a fixed-size message in the buffer
sl@0
  1090
	@param a copy of the message
sl@0
  1091
	@param aIndex the ordinal position of the message
sl@0
  1092
	@panic if the index is out of bounds
sl@0
  1093
	@panic the message specified is not of the correct type
sl@0
  1094
*/	{
sl@0
  1095
	__ASSERT_COMPILE(sizeof(TWsGraphicMsgFixedBase) == (sizeof(TInt32)*2));
sl@0
  1096
	const TInt KHeaderSize = sizeof(TWsGraphicMsgFixedBase);
sl@0
  1097
	const TInt length = Length();
sl@0
  1098
	TInt ofs = 0, count = 0;
sl@0
  1099
	while(ofs < length)
sl@0
  1100
		{
sl@0
  1101
		if(count == aIndex)
sl@0
  1102
			{
sl@0
  1103
			if((TUid::Uid(IntAt(ofs)) != aMsg.TypeId()) ||
sl@0
  1104
				(IntAt(ofs+sizeof(TInt)) != aMsg.Size()) ||
sl@0
  1105
				((ofs + KHeaderSize + aMsg.Size()) > length))
sl@0
  1106
				{
sl@0
  1107
				Panic(EW32PanicGraphicBadBuffer);
sl@0
  1108
				}
sl@0
  1109
			memcpy(&aMsg,(Ptr()+ofs),KHeaderSize + aMsg.Size());
sl@0
  1110
			return;
sl@0
  1111
			}
sl@0
  1112
		count++;
sl@0
  1113
		ofs += IntAt(ofs+sizeof(TInt)) + (sizeof(TInt)*2);
sl@0
  1114
		}
sl@0
  1115
	Panic(EW32PanicGraphicBadBuffer);
sl@0
  1116
	}
sl@0
  1117