os/graphics/windowing/windowserver/nga/SERVER/openwfc/redrawmsgwindow.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) 1995-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
// Window redraw code, three sorts of redrawing are supported
sl@0
    15
// CRedrawMsgWindow handles it via sending a redraw message to the client
sl@0
    16
// 
sl@0
    17
//
sl@0
    18
sl@0
    19
#include "redrawmsgwindow.h"
sl@0
    20
#include "gc.h"
sl@0
    21
#include "playbackgc.h"
sl@0
    22
#include "inifile.h"
sl@0
    23
#include "rootwin.h"
sl@0
    24
#include "wstop.h"
sl@0
    25
#include "ANIM.H"
sl@0
    26
#include "EVQUEUE.H"
sl@0
    27
#include <s32mem.h>
sl@0
    28
#include <gdi.h>
sl@0
    29
#include "panics.h"
sl@0
    30
#include "rootwin.h"
sl@0
    31
#include "EVENT.H"
sl@0
    32
#include "wsfont.h"
sl@0
    33
#include <graphics/WSGRAPHICDRAWERINTERFACE.H>
sl@0
    34
#include <graphics/surface.h>
sl@0
    35
#include "windowelementset.h"
sl@0
    36
sl@0
    37
#include "drawresource.h"
sl@0
    38
#include "../debuglog/DEBUGLOG.H"
sl@0
    39
sl@0
    40
const TUint KDrawBufferGranularity = 240;
sl@0
    41
const TInt KRedrawRegionGranularity = 8;
sl@0
    42
const TInt KReadBufferMaxLen=0x100;
sl@0
    43
const TInt KRegionCompressThreshold = 6; // number of rectangles in region at which we try to compress it
sl@0
    44
sl@0
    45
TBool CWsRedrawMsgWindow::iAtomic = EFalse;
sl@0
    46
sl@0
    47
extern CDebugLogBase *wsDebugLog;
sl@0
    48
sl@0
    49
#ifndef _DEBUG
sl@0
    50
sl@0
    51
#define LOG_WINDOW_DRAW_COMMANDS_START(wswin,region)
sl@0
    52
#define LOG_WINDOW_DRAW_COMMANDS_END(wswin)
sl@0
    53
#define LOG_REDRAW_SEGMENT(segmentIndex,segmentType)
sl@0
    54
#define LOG_REDRAW_SEGMENT_REGION(region)
sl@0
    55
#define LOG_PLAYBACK_GC_COMMAND(opcode,data)
sl@0
    56
sl@0
    57
#else
sl@0
    58
sl@0
    59
#define LOG_WINDOW_DRAW_COMMANDS_START(wswin,region) LogDrawCommandsStart(wswin,region)
sl@0
    60
#define LOG_WINDOW_DRAW_COMMANDS_END(wswin) LogDrawCommandsEnd(wswin)
sl@0
    61
#define LOG_REDRAW_SEGMENT(segmentIndex,segmentType) LogRedrawSegment(segmentIndex, segmentType)
sl@0
    62
#define LOG_REDRAW_SEGMENT_REGION(region) {if(wsDebugLog){ LogRegion(region);}}
sl@0
    63
#define LOG_PLAYBACK_GC_COMMAND(opcode,data)    {if (wsDebugLog) {wsDebugLog->Command(WS_HANDLE_GC, opcode, data, NULL);}}
sl@0
    64
sl@0
    65
LOCAL_C void LogRedrawSegment(TUint aSegmentIndex, CWsRedrawMsgWindow::TRedrawSegmentType aSegmentType)
sl@0
    66
	{
sl@0
    67
	if (wsDebugLog)
sl@0
    68
		{
sl@0
    69
		TBuf<LogTBufSize> log;
sl@0
    70
		TTruncateOverflow overflow;
sl@0
    71
		_LIT(KLogRedrawSegment, ">> CRedrawSegment[%d] ");
sl@0
    72
		log.AppendFormat(KLogRedrawSegment, &overflow, aSegmentIndex);
sl@0
    73
		_LIT(KLogRedrawSegmentPending, "Pending");
sl@0
    74
		_LIT(KLogRedrawSegmentRedraw, "Redraw");
sl@0
    75
		switch(aSegmentType)
sl@0
    76
			{
sl@0
    77
			case CWsRedrawMsgWindow::ESegmentTypePendingRedraw :
sl@0
    78
				log.AppendFormat(KLogRedrawSegmentPending, &overflow);
sl@0
    79
				break;
sl@0
    80
			case CWsRedrawMsgWindow::ESegmentTypeRedraw :
sl@0
    81
				log.AppendFormat(KLogRedrawSegmentRedraw, &overflow);
sl@0
    82
				break;
sl@0
    83
			default :
sl@0
    84
				{
sl@0
    85
				}
sl@0
    86
			}
sl@0
    87
		wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
sl@0
    88
		}
sl@0
    89
	}
sl@0
    90
sl@0
    91
LOCAL_C void LogDrawCommandsStart(const CWsWindow* aWsWin, const TRegion* aRegion)
sl@0
    92
	{
sl@0
    93
	if (wsDebugLog)
sl@0
    94
		{
sl@0
    95
		_LIT(KLogDrawCommandsStart, ">> CWsRedrawMsgWindow::DrawCommandsL() [%S][app %d] RWindow[%d]");
sl@0
    96
		const TDesC& clientName = aWsWin->WsOwner()->Client().FullName();
sl@0
    97
		TBuf<LogTBufSize> log;
sl@0
    98
		TTruncateOverflow overflow;
sl@0
    99
		log.AppendFormat(KLogDrawCommandsStart, &overflow, &clientName, aWsWin->WsOwner()->ConnectionHandle(), aWsWin->LogHandle());
sl@0
   100
		wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
sl@0
   101
		LogRegion(aRegion);
sl@0
   102
		}
sl@0
   103
	}
sl@0
   104
sl@0
   105
LOCAL_C void LogDrawCommandsEnd(const CWsWindow* aWsWin)
sl@0
   106
	{
sl@0
   107
	if (wsDebugLog)
sl@0
   108
		{
sl@0
   109
		_LIT(KLogDrawCommandsEnd, "<< CWsRedrawMsgWindow::DrawCommandsL() [%S][app %d] RWindow[%d]");
sl@0
   110
		const TDesC& clientName = aWsWin->WsOwner()->Client().FullName();
sl@0
   111
		TBuf<LogTBufSize> log;
sl@0
   112
		TTruncateOverflow overflow;
sl@0
   113
		log.AppendFormat(KLogDrawCommandsEnd, &overflow, &clientName, aWsWin->WsOwner()->ConnectionHandle(), aWsWin->LogHandle());		
sl@0
   114
		wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
sl@0
   115
		}
sl@0
   116
	}
sl@0
   117
sl@0
   118
#endif
sl@0
   119
sl@0
   120
//
sl@0
   121
// Redraw windows //
sl@0
   122
//
sl@0
   123
sl@0
   124
CWsRedrawMsgWindow::CRedrawSegment::CRedrawSegment(CWsRedrawMsgWindow& aRedraw) : iRedraw(aRedraw) 
sl@0
   125
	{
sl@0
   126
	}
sl@0
   127
sl@0
   128
CWsRedrawMsgWindow::CRedrawSegment* CWsRedrawMsgWindow::CRedrawSegment::NewLC(const TRect& aRect, TRedrawSegmentType aNewRegionType, CWsRedrawMsgWindow& aRedraw) 
sl@0
   129
	{
sl@0
   130
	CRedrawSegment* self = new (ELeave) CRedrawSegment(aRedraw);
sl@0
   131
	CleanupStack::PushL(self);
sl@0
   132
	self->ConstructL(aRect, aNewRegionType);
sl@0
   133
	return self;
sl@0
   134
	}
sl@0
   135
sl@0
   136
void CWsRedrawMsgWindow::StaticInitL()
sl@0
   137
	{
sl@0
   138
	// Note that the wsini.ini setting NONREDRAWAGELIMIT is obselete in
sl@0
   139
	// Wserv2 NGA.  The effective behaviour of the system is as if the
sl@0
   140
	// value was set to zero.
sl@0
   141
	
sl@0
   142
	_LIT(KAtomicRedraws, "ATOMICREDRAWS");
sl@0
   143
	if (WsIniFile->FindVar(KAtomicRedraws) || 
sl@0
   144
			WsIniFile->FindVar(KWSERVIniFileVarChangeTracking)) //ChangeTracking requires atomic redraws to be enabled to
sl@0
   145
		iAtomic = ETrue;										//prevent playing redraw-segemts before they are completed.
sl@0
   146
	else
sl@0
   147
		iAtomic = EFalse;
sl@0
   148
	}
sl@0
   149
sl@0
   150
static TInt FindBitmapRefByHandle(const TInt* aKey, const CFbsBitmapRef& aBitmapRef)
sl@0
   151
    { // compare handles
sl@0
   152
    return *aKey - aBitmapRef.Handle();
sl@0
   153
    }
sl@0
   154
sl@0
   155
static TInt InsertBitmapRefByHandle(const CFbsBitmapRef& aFirst, const CFbsBitmapRef& aSecond)
sl@0
   156
    {
sl@0
   157
    return aFirst.Handle() - aSecond.Handle();
sl@0
   158
    }
sl@0
   159
sl@0
   160
void CWsRedrawMsgWindow::CRedrawSegment::ReleaseFontsBitmapsDrawableSources()
sl@0
   161
	{
sl@0
   162
	// release Bitmap, Font and Drawable Source handles
sl@0
   163
	TInt count = iWsBitmapArray.Count();
sl@0
   164
	TInt ii;
sl@0
   165
	for (ii = count - 1; ii >= 0; ii--)
sl@0
   166
		{
sl@0
   167
		iWsBitmapArray[ii]->DecRefCount();
sl@0
   168
		iWsBitmapArray.Remove(ii);
sl@0
   169
		}
sl@0
   170
sl@0
   171
	count = iWsFontArray.Count();
sl@0
   172
	for (ii = count - 1; ii >= 0; --ii)
sl@0
   173
		{
sl@0
   174
		CWsFontCache::Instance()->ReleaseFont(iWsFontArray[ii]);
sl@0
   175
		iWsFontArray.Remove(ii);
sl@0
   176
		}
sl@0
   177
sl@0
   178
	iRedraw.CleanupBitmapRefArray(iBitmapHandleArray);
sl@0
   179
	iBitmapHandleArray.Close();
sl@0
   180
sl@0
   181
	count = iWsDrawableSourceArray.Count();
sl@0
   182
	for(ii = count - 1 ; ii >= 0; ii--)
sl@0
   183
		{
sl@0
   184
		iWsDrawableSourceArray[ii]->DecRefCount();
sl@0
   185
		iWsDrawableSourceArray.Remove(ii);
sl@0
   186
		}
sl@0
   187
	}
sl@0
   188
sl@0
   189
/* Set new rectangle and region type for initial or reset redraw region
sl@0
   190
 @leave KErrNoMemory no memory to update region details
sl@0
   191
 */
sl@0
   192
void CWsRedrawMsgWindow::CRedrawSegment::ConstructL(const TRect& aRect, TRedrawSegmentType aNewRegionType)
sl@0
   193
	{
sl@0
   194
	iDrawCommands = CBufSeg::NewL(KDrawBufferGranularity);
sl@0
   195
	iCreationTime.UniversalTime();
sl@0
   196
sl@0
   197
	iRegion.AddRect(aRect);
sl@0
   198
	if (iRegion.CheckError())
sl@0
   199
		{
sl@0
   200
		User::Leave(KErrNoMemory);
sl@0
   201
		}
sl@0
   202
	iRedrawSegmentType = aNewRegionType;
sl@0
   203
	}
sl@0
   204
sl@0
   205
CWsRedrawMsgWindow::CRedrawSegment::~CRedrawSegment()
sl@0
   206
	{
sl@0
   207
	
sl@0
   208
	if (iDrawCommands)
sl@0
   209
		{
sl@0
   210
		delete iDrawCommands;
sl@0
   211
		}
sl@0
   212
	iRegion.Close();
sl@0
   213
	// Release Font,  Bitmap and Drawable Source handles, close arrays
sl@0
   214
	ReleaseFontsBitmapsDrawableSources();
sl@0
   215
	iWsBitmapArray.Close();
sl@0
   216
	iWsFontArray.Close();
sl@0
   217
	iDrawerArray.Close();	
sl@0
   218
	iWsDrawableSourceArray.Close();
sl@0
   219
	}
sl@0
   220
sl@0
   221
sl@0
   222
TInt CWsRedrawMsgWindow::CRedrawSegment::SizeInBytes() const
sl@0
   223
	{
sl@0
   224
	TInt size = sizeof(CWsRedrawMsgWindow::CRedrawSegment);
sl@0
   225
	size += iDrawCommands->Size();
sl@0
   226
	size += iBitmapHandleArray.Count() * sizeof(TInt);
sl@0
   227
	size += iWsBitmapArray.Count() * sizeof(DWsBitmap);
sl@0
   228
	size += iWsFontArray.Count() * sizeof(CWsFbsFont);
sl@0
   229
	size += iDrawerArray.Count() * sizeof(TGraphicDrawerId);
sl@0
   230
	size += iWsDrawableSourceArray.Count() * sizeof(CWsDrawableSource);
sl@0
   231
	return size;
sl@0
   232
	}
sl@0
   233
sl@0
   234
TBool CWsRedrawMsgWindow::CRedrawSegment::AppendNonDuplicateBitmapHandleL(TInt aHandle)
sl@0
   235
    {
sl@0
   236
    TBool handleAppend = EFalse;
sl@0
   237
    TInt indexBitmapHandle = iBitmapHandleArray.Find(aHandle);
sl@0
   238
    if (indexBitmapHandle<0)
sl@0
   239
        {
sl@0
   240
        iBitmapHandleArray.AppendL(aHandle);
sl@0
   241
        handleAppend = ETrue;
sl@0
   242
        }
sl@0
   243
    return handleAppend;
sl@0
   244
    }
sl@0
   245
sl@0
   246
CWsRedrawMsgWindow::CWsRedrawMsgWindow(CWsWindow *aWin)
sl@0
   247
	:CWsWindowRedraw(aWin), iBackColor(aWin->RootWindow()->DefaultBackgroundColor()), iFlags(EBackgroundClear|EStoringEntireWindow),
sl@0
   248
	iRedrawSegments(KRedrawRegionGranularity),
sl@0
   249
	iCurrentSegment(0),
sl@0
   250
	iOSBStatus(ETrue)
sl@0
   251
	{
sl@0
   252
	}
sl@0
   253
sl@0
   254
void CWsRedrawMsgWindow::ConstructL()
sl@0
   255
	{
sl@0
   256
	CWsWindowRedraw::ConstructL();
sl@0
   257
	Invalidate(&WsWin()->Rel());
sl@0
   258
	iWsWin->WsOwner()->RedrawQueue()->ReCalcOrder();
sl@0
   259
	}
sl@0
   260
sl@0
   261
CWsRedrawMsgWindow::~CWsRedrawMsgWindow()
sl@0
   262
	{
sl@0
   263
	WS_ASSERT_DEBUG(iMemoryLock == 0, EWsPanicMemoryLock);
sl@0
   264
	RemoveFromRedrawQueueIfEmpty();	
sl@0
   265
	iInvalid.Close();
sl@0
   266
	iLocalRedrawRegion.Close();
sl@0
   267
	iRedrawSegments.ResetAndDestroy();
sl@0
   268
	TInt count = iFbsBitmapRefArray.Count();
sl@0
   269
	WS_ASSERT_DEBUG(count==0,EWsPanicBitmapArrayNotEmpty);
sl@0
   270
	iFbsBitmapRefArray.ResetAndDestroy();
sl@0
   271
	}
sl@0
   272
sl@0
   273
/**
sl@0
   274
These three functions actually check for a value they have already asserted on.  This is intentional.
sl@0
   275
*/
sl@0
   276
void CWsRedrawMsgWindow::ExpandCommandBufferL(TInt aLength)
sl@0
   277
	{
sl@0
   278
	WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
sl@0
   279
	
sl@0
   280
	if (iCurrentSegment)
sl@0
   281
		{
sl@0
   282
		// need more space?
sl@0
   283
		if (iCurrentSegment->iCurrentCommandBufferWritePos + aLength > iCurrentSegment->iDrawCommands->Size())
sl@0
   284
			{
sl@0
   285
			iCurrentSegment->iDrawCommands->ResizeL(iCurrentSegment->iCurrentCommandBufferWritePos + aLength);
sl@0
   286
			}
sl@0
   287
		}
sl@0
   288
	}
sl@0
   289
sl@0
   290
void CWsRedrawMsgWindow::CommandBufferWrite(const TDesC8& aDes, TInt aLength)
sl@0
   291
	{
sl@0
   292
	WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
sl@0
   293
	WS_ASSERT_DEBUG(iCurrentSegment->iCurrentCommandBufferWritePos + aLength <= iCurrentSegment->iDrawCommands->Size(), EWsPanicDrawCommandsInvalidState);
sl@0
   294
	if (iCurrentSegment)
sl@0
   295
		{
sl@0
   296
		iCurrentSegment->iDrawCommands->Write(iCurrentSegment->iCurrentCommandBufferWritePos, aDes, aLength);
sl@0
   297
		iCurrentSegment->iCurrentCommandBufferWritePos += aLength;
sl@0
   298
		}
sl@0
   299
	}
sl@0
   300
sl@0
   301
void CWsRedrawMsgWindow::CommandBufferWrite(const TAny* aPtr,TInt aLength)
sl@0
   302
	{
sl@0
   303
	WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
sl@0
   304
	WS_ASSERT_DEBUG(iCurrentSegment->iCurrentCommandBufferWritePos + aLength <= iCurrentSegment->iDrawCommands->Size(), EWsPanicDrawCommandsInvalidState);
sl@0
   305
	if (iCurrentSegment)
sl@0
   306
		{
sl@0
   307
		iCurrentSegment->iDrawCommands->Write(iCurrentSegment->iCurrentCommandBufferWritePos, aPtr, aLength);
sl@0
   308
		iCurrentSegment->iCurrentCommandBufferWritePos += aLength;
sl@0
   309
		}
sl@0
   310
	}
sl@0
   311
sl@0
   312
/*------------------------------------------------------------------------------
sl@0
   313
  Description: Processes draw commands. These are received as opcodes.
sl@0
   314
 -----------------------------------------------------------------------------*/
sl@0
   315
TBool CWsRedrawMsgWindow::CommandL(TInt aOpcode, TWsWinCmdUnion &aCmd)
sl@0
   316
	{
sl@0
   317
	switch(aOpcode)
sl@0
   318
		{
sl@0
   319
		case EWsWinOpEnableOSB:
sl@0
   320
			iOSBStatus = ETrue;
sl@0
   321
			break;
sl@0
   322
		case EWsWinOpDisableOSB:
sl@0
   323
			iOSBStatus = EFalse;
sl@0
   324
			break;
sl@0
   325
		case EWsWinOpSetBackgroundColor:
sl@0
   326
			iBackColor = *aCmd.rgb;
sl@0
   327
			iFlags |= EBackgroundClear;
sl@0
   328
			break;
sl@0
   329
		case EWsWinOpSetNoBackgroundColor:
sl@0
   330
			iFlags &= ~EBackgroundClear;
sl@0
   331
			break;
sl@0
   332
		case EWsWinOpInvalidate:
sl@0
   333
			Invalidate(aCmd.rect);
sl@0
   334
			break;
sl@0
   335
		case EWsWinOpInvalidateFull:
sl@0
   336
			Invalidate();
sl@0
   337
			break;
sl@0
   338
		case EWsWinOpBeginRedraw:
sl@0
   339
			BeginRedraw(aCmd.rect);
sl@0
   340
			ValidateRect(aCmd.rect);
sl@0
   341
			break;
sl@0
   342
		case EWsWinOpBeginRedrawFull:
sl@0
   343
			BeginRedraw(NULL);
sl@0
   344
			ValidateRect(NULL);
sl@0
   345
			break;
sl@0
   346
		case EWsWinOpEndRedraw:
sl@0
   347
			EndRedraw();
sl@0
   348
			break;
sl@0
   349
		case EWsWinOpGetInvalidRegionCount:
sl@0
   350
			{
sl@0
   351
			SetReply(iInvalid.Count());
sl@0
   352
			}
sl@0
   353
			break;
sl@0
   354
		case EWsWinOpGetInvalidRegion:
sl@0
   355
			{
sl@0
   356
			if ((*aCmd.Int) <= 0)
sl@0
   357
				OwnerPanic(EWservPanicInvalidRegionCount);
sl@0
   358
			if ((!iInvalid.CheckError()) && iInvalid.Count() == (*aCmd.Int))
sl@0
   359
				{
sl@0
   360
				CWsClient::ReplyBuf(iInvalid.RectangleList(),(*aCmd.Int) * sizeof(TRect));
sl@0
   361
				SetReply(EFalse);
sl@0
   362
				}
sl@0
   363
			else
sl@0
   364
				SetReply(ETrue);
sl@0
   365
			}
sl@0
   366
			break;
sl@0
   367
		case EWsWinOpStoreDrawCommands:
sl@0
   368
			/* If the client asks us not to store commands, we still store the commands
sl@0
   369
			for the region of the window which can be seen through the parent, but
sl@0
   370
			won't attempt to obtain the entire window.
sl@0
   371
			*/
sl@0
   372
			if (*aCmd.Bool)
sl@0
   373
				{
sl@0
   374
				SetScope(EStoreEntireWindow);
sl@0
   375
				}
sl@0
   376
			else
sl@0
   377
				{
sl@0
   378
				// Clients that turn their redraw store off will still get one,
sl@0
   379
				// but it will only attempt to store the current viewport.
sl@0
   380
				SetScope(EStoreViewport);
sl@0
   381
				}
sl@0
   382
			break;
sl@0
   383
		case EWsWinOpHandleTransparencyUpdate:	// deprecated
sl@0
   384
		case EWsWinOpSetTransparencyBitmap:		// deprecated
sl@0
   385
		case EWsWinOpSetTransparencyFactor:		// deprecated
sl@0
   386
		case EWsWinOpSetTransparencyBitmapCWs:	// deprecated
sl@0
   387
			break;	// do nothing.
sl@0
   388
		case EWsWinOpIsRedrawStoreEnabled:
sl@0
   389
			SetReply(ETrue);
sl@0
   390
			break;
sl@0
   391
		case EWsWinOpClearRedrawStore:
sl@0
   392
			DiscardStoredCommands();
sl@0
   393
			break;
sl@0
   394
		case EWsWinOpSetBackgroundSurface:
sl@0
   395
			SetBackgroundSurfaceL(*aCmd.Surface);
sl@0
   396
			break;
sl@0
   397
		case EWsWinOpSetBackgroundSurfaceConfig:
sl@0
   398
			SetBackgroundSurfaceL(aCmd.SurfaceConfigurationAndTrigger->surfaceConfig, aCmd.SurfaceConfigurationAndTrigger->triggerRedraw, EFalse);
sl@0
   399
			break;
sl@0
   400
		case EWsWinOpRemoveBackgroundSurface:
sl@0
   401
			RemoveBackgroundSurface(*aCmd.Bool);
sl@0
   402
			break;
sl@0
   403
		case EWsWinOpGetBackgroundSurfaceConfig:
sl@0
   404
			{
sl@0
   405
			TSurfaceConfiguration tempConfiguration = *aCmd.SurfaceConfiguration;
sl@0
   406
			GetBackgroundSurfaceL(tempConfiguration);
sl@0
   407
			TInt tempSize = aCmd.SurfaceConfiguration->Size();
sl@0
   408
			if (sizeof(TSurfaceConfiguration)<tempSize)
sl@0
   409
				tempSize = sizeof(TSurfaceConfiguration);
sl@0
   410
			CWsClient::ReplyBuf(&tempConfiguration,tempSize);
sl@0
   411
			}
sl@0
   412
			break;
sl@0
   413
		default:
sl@0
   414
			return(EFalse);
sl@0
   415
		}
sl@0
   416
	return(ETrue);
sl@0
   417
	}
sl@0
   418
sl@0
   419
/**
sl@0
   420
*/
sl@0
   421
void CWsRedrawMsgWindow::BeginRedraw(const TRect* aRect)
sl@0
   422
	{
sl@0
   423
	if(InRedraw())
sl@0
   424
		OwnerPanic(EWservPanicDrawCommandsInvalidState);
sl@0
   425
	iFlags|=EBeginEndRedraw;
sl@0
   426
	TRAPD(err,DoBeginRedrawL(aRect));
sl@0
   427
	DiscardStoredCommandsIfError(err);
sl@0
   428
	}
sl@0
   429
sl@0
   430
void CWsRedrawMsgWindow::DoBeginRedrawL(const TRect* aRect)
sl@0
   431
	{
sl@0
   432
	const TRect redrawRect = (aRect ? *aRect : TRect(WsWin()->Size()));
sl@0
   433
	if (redrawRect.IsEmpty())
sl@0
   434
		{
sl@0
   435
		//Skip empty rects since they are not added to the region
sl@0
   436
		iCurrentSegment = NULL;
sl@0
   437
		}
sl@0
   438
	else
sl@0
   439
		{
sl@0
   440
		CreateNewSegmentL(redrawRect, CWsRedrawMsgWindow::ESegmentTypePendingRedraw);
sl@0
   441
		if (!iAtomic)
sl@0
   442
			PromotePendingSegment();
sl@0
   443
		}
sl@0
   444
	}
sl@0
   445
sl@0
   446
void CWsRedrawMsgWindow::Invalidate(const TRect * aRect)
sl@0
   447
	{
sl@0
   448
	//The memory allocation in this function can trigger a call to ReleaseMemory(), which would
sl@0
   449
	//recursively call this function again. This would cause a memory leak. To avoid this
sl@0
   450
	//we call Lock() to block ReleaseMemory() from this object while executing this function.
sl@0
   451
    Lock();
sl@0
   452
	if (!aRect)
sl@0
   453
		{
sl@0
   454
		iInvalid.Clear();
sl@0
   455
		iInvalid.Copy(iWsWin->WindowArea());
sl@0
   456
		iInvalid.Offset(-iWsWin->Origin());
sl@0
   457
		}
sl@0
   458
	else if((!aRect->IsEmpty()) && aRect->IsNormalized())
sl@0
   459
		{
sl@0
   460
		iInvalid.AddRect(*aRect);
sl@0
   461
		iInvalid.Tidy();
sl@0
   462
		}
sl@0
   463
	if (iWsWin->IsVisible())
sl@0
   464
		{
sl@0
   465
		QueueRedraw();
sl@0
   466
		iWsWin->WsOwner()->TriggerRedraw(); //wtf isn't the redrawq already scheduling itself?
sl@0
   467
		}
sl@0
   468
    Unlock();
sl@0
   469
	}
sl@0
   470
sl@0
   471
/**
sl@0
   472
Obtains a region from the redraw store, intersects it with the global redraw region which
sl@0
   473
we have been asked to draw to, and returns the intersection.
sl@0
   474
*/
sl@0
   475
const TRegion * CWsRedrawMsgWindow::ReadRegion(const TInt aRegionNum)
sl@0
   476
	{
sl@0
   477
	// We are drawing to the global region, and we have a region in the redraw store to clip to.
sl@0
   478
	// We want the intersection of these to actually draw to.
sl@0
   479
	iLocalRedrawRegion.Copy(iRedrawSegments[aRegionNum]->iRegion);
sl@0
   480
	iLocalRedrawRegion.Offset(WsWin()->Origin());
sl@0
   481
	iLocalRedrawRegion.Intersect(*iRedrawRegion);
sl@0
   482
	iLocalRedrawRegion.Tidy();
sl@0
   483
sl@0
   484
	// If the resulting region is empty there is no point drawing its corresponding commands
sl@0
   485
	if (iLocalRedrawRegion.IsEmpty())
sl@0
   486
		return NULL;
sl@0
   487
	else
sl@0
   488
		return &iLocalRedrawRegion;
sl@0
   489
	}
sl@0
   490
sl@0
   491
void CWsRedrawMsgWindow::SubtractRectFromSegmentArray(const TRect& aRect)
sl@0
   492
	{
sl@0
   493
	for (TInt regionNum = iRedrawSegments.Count() - 1; regionNum >= 0; --regionNum)
sl@0
   494
		{
sl@0
   495
		if (iRedrawSegments[regionNum]->iRedrawSegmentType != ESegmentTypePendingRedraw)
sl@0
   496
			{
sl@0
   497
			RWsRegion& region = iRedrawSegments[regionNum]->iRegion;
sl@0
   498
			region.SubRect(aRect);
sl@0
   499
			if (region.CheckError())
sl@0
   500
				{
sl@0
   501
				// Ouch. Drop the now broken segment and ask for a full redraw
sl@0
   502
				delete iRedrawSegments[regionNum];
sl@0
   503
				iRedrawSegments.Remove(regionNum);
sl@0
   504
				Invalidate();
sl@0
   505
				}
sl@0
   506
			else
sl@0
   507
				{
sl@0
   508
				// check if region has zero uncovered rectangles left
sl@0
   509
				if (region.IsEmpty())
sl@0
   510
					{ // delete draw commands, release bitmaps and fonts
sl@0
   511
					delete iRedrawSegments[regionNum];
sl@0
   512
					iRedrawSegments.Remove(regionNum);
sl@0
   513
					}
sl@0
   514
				else
sl@0
   515
					{
sl@0
   516
					if (region.Count() > KRegionCompressThreshold)
sl@0
   517
						{ // tidy up the rectangles
sl@0
   518
						region.Tidy();
sl@0
   519
						}
sl@0
   520
					}
sl@0
   521
				}
sl@0
   522
			}
sl@0
   523
		}
sl@0
   524
	// coverity[extend_simple_error]	
sl@0
   525
	}
sl@0
   526
sl@0
   527
/*------------------------------------------------------------------------------
sl@0
   528
  Description: Clears out the command buffer if aError indicates an
sl@0
   529
               error has occurred whilst storing commands.
sl@0
   530
 -----------------------------------------------------------------------------*/
sl@0
   531
void CWsRedrawMsgWindow::DiscardStoredCommandsIfError(TInt aError)
sl@0
   532
	{
sl@0
   533
	if (aError != KErrNone)
sl@0
   534
		{
sl@0
   535
		// Discard stored commands by clearing out the command buffer
sl@0
   536
		DiscardStoredCommands();
sl@0
   537
sl@0
   538
		if(!(iFlags&ECurrentRedrawFailedStorage)) //first time we fail during the current redraw
sl@0
   539
			{
sl@0
   540
			iFlags |= ECurrentRedrawFailedStorage;
sl@0
   541
sl@0
   542
			if(!(iFlags&EPreviousRedrawFailedStorage)) //unless the previous redraw failed as well (to avoid infinite loop)
sl@0
   543
				{
sl@0
   544
				Invalidate(); //request new redraw from the client
sl@0
   545
				}
sl@0
   546
			}
sl@0
   547
		}
sl@0
   548
	}
sl@0
   549
sl@0
   550
/*------------------------------------------------------------------------------
sl@0
   551
  Description: If the graphics context has changed and we are currently storing
sl@0
   552
               commands, store the data given by aCmdData.
sl@0
   553
               
sl@0
   554
 -----------------------------------------------------------------------------*/
sl@0
   555
TBool CWsRedrawMsgWindow::DrawCommand(CWsGc* aGc,const TAny *aCmdData)
sl@0
   556
	{
sl@0
   557
	// Store commands, clearing out buffer if error occurs.
sl@0
   558
	TRAPD(err,StoreDrawCommandL(aGc,aCmdData));
sl@0
   559
	DiscardStoredCommandsIfError(err);
sl@0
   560
	return EFalse;
sl@0
   561
	}
sl@0
   562
sl@0
   563
void CWsRedrawMsgWindow::GcAttributeChange(CWsGc* aGc,const TAny *aCmdData)
sl@0
   564
	{
sl@0
   565
	if (iLastDrawGc == aGc && iCurrentSegment)
sl@0
   566
		{
sl@0
   567
		TInt err = KErrNone;
sl@0
   568
		if (IsWsFontOperation(CWsClient::iCurrentCommand.iOpcode))
sl@0
   569
			{
sl@0
   570
			TWsGcCmdUnion pData;
sl@0
   571
			pData.any=aCmdData;
sl@0
   572
			TRAP(err,AddWsFontL(*pData.UInt));
sl@0
   573
			}
sl@0
   574
		if (KErrNone == err)
sl@0
   575
			{
sl@0
   576
			TRAP(err,AppendCommandL(aCmdData));
sl@0
   577
			}
sl@0
   578
		DiscardStoredCommandsIfError(err);
sl@0
   579
		}
sl@0
   580
	
sl@0
   581
	// INC135845: 
sl@0
   582
	// Retain the bitmap handle for the lifetime of the redraw store
sl@0
   583
	// If the client destroys it, we will still have a reference to it
sl@0
   584
	if (iCurrentSegment && CWsClient::iCurrentCommand.iOpcode == EWsGcOpUseBrushPattern)
sl@0
   585
		{
sl@0
   586
		TInt err = KErrNone;
sl@0
   587
		TWsGcCmdUnion pData;
sl@0
   588
		pData.any=aCmdData;
sl@0
   589
		TRAP(err, AddFbsBitmapsL(*pData.handle, 0));
sl@0
   590
		DiscardStoredCommandsIfError(err);
sl@0
   591
		}
sl@0
   592
	}
sl@0
   593
sl@0
   594
void CWsRedrawMsgWindow::GcDeactivate(CWsGc* aGc)
sl@0
   595
	{
sl@0
   596
	if (iLastDrawGc==aGc)
sl@0
   597
		iLastDrawGc=NULL;
sl@0
   598
	}
sl@0
   599
sl@0
   600
inline TBool CWsRedrawMsgWindow::IsFbsBitmapOperation(TInt aOpCode) const
sl@0
   601
	{
sl@0
   602
	WS_ASSERT_DEBUG((EWsGcOpGdiBlt3==EWsGcOpGdiBlt2+1)&&(EWsGcOpGdiBltMasked==EWsGcOpGdiBlt3+1)
sl@0
   603
				&&(EWsGcOpDrawBitmap2==EWsGcOpDrawBitmap+1)&&(EWsGcOpDrawBitmap3==EWsGcOpDrawBitmap2+1)&&(EWsGcOpDrawBitmapMasked==EWsGcOpDrawBitmap3+1),EWsPanicBitmapOpcodeInvalid);
sl@0
   604
	return (aOpCode>=EWsGcOpGdiBlt2&&aOpCode<=EWsGcOpGdiBltMasked)||(aOpCode>=EWsGcOpDrawBitmap&&aOpCode<=EWsGcOpDrawBitmapMasked)||(aOpCode==EWsGcOpGdiAlphaBlendBitmaps);
sl@0
   605
	}
sl@0
   606
sl@0
   607
inline TBool CWsRedrawMsgWindow::IsWsBitmapOperation(TInt aOpCode) const
sl@0
   608
	{
sl@0
   609
	WS_ASSERT_DEBUG((EWsGcOpGdiBlt3==EWsGcOpGdiBlt2+1)&&(EWsGcOpGdiBltMasked==EWsGcOpGdiBlt3+1)
sl@0
   610
				&&(EWsGcOpDrawBitmap2==EWsGcOpDrawBitmap+1)&&(EWsGcOpDrawBitmap3==EWsGcOpDrawBitmap2+1)&&(EWsGcOpDrawBitmapMasked==EWsGcOpDrawBitmap3+1),EWsPanicBitmapOpcodeInvalid);
sl@0
   611
	return (aOpCode>=EWsGcOpGdiWsBlt2&&aOpCode<=EWsGcOpGdiWsBltMasked)||(aOpCode==EWsGcOpGdiWsAlphaBlendBitmaps)||(aOpCode==EWsGcOpWsDrawBitmapMasked);
sl@0
   612
	}
sl@0
   613
sl@0
   614
inline TBool CWsRedrawMsgWindow::IsWsFontOperation(TInt aOpCode) const
sl@0
   615
	{
sl@0
   616
	return aOpCode==EWsGcOpUseFont;
sl@0
   617
	}
sl@0
   618
sl@0
   619
inline TBool CWsRedrawMsgWindow::IsDrawWsGraphicOperation(TInt aOpCode) const
sl@0
   620
	{
sl@0
   621
	return (aOpCode == EWsGcOpDrawWsGraphic) || (aOpCode == EWsGcOpDrawWsGraphicPtr);
sl@0
   622
	}
sl@0
   623
sl@0
   624
inline TBool CWsRedrawMsgWindow::IsWsDrawableSourceOperation(TInt aOpCode) const
sl@0
   625
	{
sl@0
   626
	return (aOpCode == EWsGcOpDrawResourceToPos) || (aOpCode == EWsGcOpDrawResourceToRect) || (aOpCode == EWsGcOpDrawResourceFromRectToRect) || (aOpCode == EWsGcOpDrawResourceWithData);
sl@0
   627
	}
sl@0
   628
sl@0
   629
void CWsRedrawMsgWindow::ReplaceAndAppendCommandL(TInt aOpcode,const TAny* aCmdData)
sl@0
   630
	{
sl@0
   631
	const TInt KCharWidthInBytes = 2; // # of bytes for a Unicode character
sl@0
   632
	CWsClient* owner=iWsWin->WsOwner();
sl@0
   633
	WS_ASSERT_DEBUG(owner,EWsPanicDrawCommandsNullSession);
sl@0
   634
sl@0
   635
	// aCmdData doesn't contain data, it should be retrieved from client space using remote read
sl@0
   636
	TWsGcCmdUnion cmd;
sl@0
   637
	cmd.any=aCmdData;
sl@0
   638
	TUint16 newOpcode=EWsGcOpDrawText;
sl@0
   639
	TInt strLen=0;
sl@0
   640
	switch (aOpcode)
sl@0
   641
		{
sl@0
   642
		case EWsGcOpDrawTextPtr:
sl@0
   643
			newOpcode=EWsGcOpDrawText;
sl@0
   644
			strLen=cmd.DrawText->length;
sl@0
   645
			break;
sl@0
   646
		case EWsGcOpDrawTextVerticalPtr:
sl@0
   647
			newOpcode=EWsGcOpDrawTextVertical;
sl@0
   648
			strLen=cmd.DrawTextVertical->length;
sl@0
   649
			break;
sl@0
   650
		case EWsGcOpDrawBoxTextPtr:
sl@0
   651
			newOpcode=EWsGcOpDrawBoxText;
sl@0
   652
			strLen=cmd.BoxText->length;
sl@0
   653
			break;
sl@0
   654
		case EWsGcOpDrawBoxTextVerticalPtr:
sl@0
   655
			newOpcode=EWsGcOpDrawBoxTextVertical;
sl@0
   656
			strLen=cmd.DrawBoxTextVertical->length;
sl@0
   657
 			break;
sl@0
   658
		case EWsGcOpDrawTextInContextPtr:
sl@0
   659
			newOpcode=EWsGcOpDrawTextInContextPtr1;
sl@0
   660
			strLen=cmd.DrawTextInContext->length;
sl@0
   661
			break;
sl@0
   662
		case EWsGcOpDrawTextInContextVerticalPtr:
sl@0
   663
			newOpcode=EWsGcOpDrawTextInContextVerticalPtr1;
sl@0
   664
			strLen=cmd.DrawTextInContextVertical->length;
sl@0
   665
			break;
sl@0
   666
		case EWsGcOpDrawBoxTextInContextPtr:
sl@0
   667
			newOpcode=EWsGcOpDrawBoxTextInContextPtr1;
sl@0
   668
			strLen=cmd.BoxTextInContext->length;
sl@0
   669
			break;
sl@0
   670
		case EWsGcOpDrawBoxTextInContextVerticalPtr:
sl@0
   671
			newOpcode=EWsGcOpDrawBoxTextInContextVerticalPtr1;
sl@0
   672
			strLen=cmd.DrawBoxTextInContextVertical->length;
sl@0
   673
			break;
sl@0
   674
		}
sl@0
   675
	TInt strSize = strLen * KCharWidthInBytes;
sl@0
   676
	TInt oldCmdLen=CWsClient::iCurrentCommand.iCmdLength;
sl@0
   677
	TInt newCmdLen=sizeof(TWsCmdHeaderBase)+oldCmdLen+strSize;
sl@0
   678
	// resize buffer
sl@0
   679
	ExpandCommandBufferL(newCmdLen);
sl@0
   680
	// update current command to reflect the new command and data
sl@0
   681
	CWsClient::iCurrentCommand.iOpcode=newOpcode;
sl@0
   682
	CWsClient::iCurrentCommand.iOpcode|=EWsGcOpFlagDrawOp;
sl@0
   683
	CWsClient::iCurrentCommand.iCmdLength=(TInt16)(oldCmdLen+strSize);
sl@0
   684
	// write command header
sl@0
   685
	CommandBufferWrite(&CWsClient::iCurrentCommand, sizeof(TWsCmdHeaderBase));
sl@0
   686
	// write command
sl@0
   687
	CommandBufferWrite(aCmdData, oldCmdLen);
sl@0
   688
sl@0
   689
	// remote read
sl@0
   690
	TBuf<KReadBufferMaxLen> buf;
sl@0
   691
	TInt len=KReadBufferMaxLen;
sl@0
   692
	TInt bufOffset=0;
sl@0
   693
	TInt toGo=strLen;
sl@0
   694
	while(toGo>0)
sl@0
   695
		{
sl@0
   696
		if (len>toGo)
sl@0
   697
			len=toGo;
sl@0
   698
		owner->RemoteRead(buf,bufOffset);
sl@0
   699
		CommandBufferWrite(buf.Ptr(), len * KCharWidthInBytes);
sl@0
   700
		bufOffset+=len;
sl@0
   701
		toGo-=len;
sl@0
   702
		}
sl@0
   703
	}
sl@0
   704
sl@0
   705
/*------------------------------------------------------------------------------
sl@0
   706
  Description: Stores drawing related commands into the command buffer
sl@0
   707
 -----------------------------------------------------------------------------*/
sl@0
   708
void CWsRedrawMsgWindow::StoreDrawCommandL(CWsGc* aGc,const TAny *aCmdData)
sl@0
   709
	{
sl@0
   710
	TWsGcOpcodes currentOpcode = static_cast<TWsGcOpcodes>(CWsClient::iCurrentCommand.iOpcode);
sl@0
   711
	
sl@0
   712
	if (!InRedraw())
sl@0
   713
		{
sl@0
   714
		// We've received a drawing operation outside of a BeginRedraw()/EndRedraw()
sl@0
   715
		// bracketed block.  This is known as a "non redraw drawing operation".
sl@0
   716
		
sl@0
   717
#ifdef __WINS__	
sl@0
   718
		// If configured, panic on non redraw drawing operations.	
sl@0
   719
		if( CWsClient::DebugEnforceRedrawCallingConvention())
sl@0
   720
			CWsClient::PanicCurrentClient(EWservPanicWindowBeginRedrawNotCalled);		
sl@0
   721
#endif
sl@0
   722
		
sl@0
   723
		// We get here if we are not configured to panic when a non-redraw drawing operation
sl@0
   724
		// is issued. 
sl@0
   725
		// We ignore this DrawOp as WServ2 is not supporting non-redraw drawing because 
sl@0
   726
		// storing such non-redraw commands in the redraw store consumes significant 
sl@0
   727
		// amounts of memory, and reduces rendering performance.
sl@0
   728
		// 
sl@0
   729
		// Issuing non-redraw drawing from a client Redrawer is a special case.  Here
sl@0
   730
		// an infinite loop would occur if we mark the area as invalid.  Therefore
sl@0
   731
		// we must always panic in this case.
sl@0
   732
		if (iWsWin->WsOwner()->ClientProcessingRedrawEvent())
sl@0
   733
			{
sl@0
   734
			// Non-redraw drawing in the Redrawer!!
sl@0
   735
			CWsClient::PanicCurrentClient(EWservPanicWindowBeginRedrawNotCalled);
sl@0
   736
			}
sl@0
   737
		else
sl@0
   738
			{
sl@0
   739
			// The Redrawer will eventually fix up the screen
sl@0
   740
			Invalidate();
sl@0
   741
			}
sl@0
   742
		return;
sl@0
   743
		}
sl@0
   744
	
sl@0
   745
	// If there is no current segment then we have discarded it at some point
sl@0
   746
	// since beginning this redraw.  
sl@0
   747
	if (iCurrentSegment)
sl@0
   748
		{
sl@0
   749
		TWsGcCmdUnion pData;
sl@0
   750
		pData.any = aCmdData;
sl@0
   751
		if (IsFbsBitmapOperation(currentOpcode))
sl@0
   752
			{
sl@0
   753
			TInt maskHandle = 0;
sl@0
   754
			TInt handle = aGc->FbsBitmapHandle(currentOpcode, pData, maskHandle);
sl@0
   755
			AddFbsBitmapsL(handle, maskHandle);
sl@0
   756
			}
sl@0
   757
		else if (IsWsBitmapOperation(currentOpcode))
sl@0
   758
			{
sl@0
   759
			TInt maskHandle = 0;
sl@0
   760
			TInt handle = aGc->WsBitmapHandle(currentOpcode, pData, maskHandle);
sl@0
   761
			AddWsBitmapsL(handle, maskHandle);
sl@0
   762
			}
sl@0
   763
		else if (IsDrawWsGraphicOperation(currentOpcode))
sl@0
   764
			{
sl@0
   765
			TGraphicDrawerId drawerId;
sl@0
   766
			drawerId.iId = pData.WsGraphic->iId;
sl@0
   767
			drawerId.iIsUid = (pData.WsGraphic->iFlags & EWsGraphicIdUid);
sl@0
   768
			iCurrentSegment->AddDrawerL(drawerId);
sl@0
   769
			}
sl@0
   770
		else if (IsWsDrawableSourceOperation(currentOpcode))
sl@0
   771
			{
sl@0
   772
			TInt handle = aGc->WsDrawableSourceHandle(currentOpcode, pData);
sl@0
   773
			AddWsDrawableSourceL(handle);
sl@0
   774
			}
sl@0
   775
sl@0
   776
		// If the graphics context has changed since last time store the new graphics
sl@0
   777
		// context attributes.
sl@0
   778
		if (aGc != iLastDrawGc)
sl@0
   779
			{
sl@0
   780
			StoreAllGcAttributesL(aGc);
sl@0
   781
			iLastDrawGc = aGc;
sl@0
   782
			}
sl@0
   783
sl@0
   784
		// For operation which requires remote read from client space, we must retrieve that data and store
sl@0
   785
		// it in command buffer at server side and change opcode if necessary e.g EWsGcOpDrawTextPtr to EWsGcOpDrawText
sl@0
   786
		// to avoid remote read during DoDrawing operation
sl@0
   787
		if (IsRemoteReadRequired(currentOpcode))
sl@0
   788
			ReplaceAndAppendCommandL(currentOpcode,aCmdData);
sl@0
   789
		else
sl@0
   790
			// Append the command data to the command buffer
sl@0
   791
			AppendCommandL(aCmdData, EWsGcOpFlagDrawOp);
sl@0
   792
		}
sl@0
   793
	}
sl@0
   794
sl@0
   795
/*------------------------------------------------------------------------------
sl@0
   796
  Description: Stores given drawing command data into the command buffer.
sl@0
   797
 -----------------------------------------------------------------------------*/
sl@0
   798
void CWsRedrawMsgWindow::AppendCommandL(const TAny* aCmdData, const TUint16 aOpcodeFlags)
sl@0
   799
	{
sl@0
   800
	if (CWsClient::iCurrentCommand.iOpcode == EWsGcOpSetClippingRegion)
sl@0
   801
		{
sl@0
   802
		// The client is defining a clipping region
sl@0
   803
sl@0
   804
		// make room for the header
sl@0
   805
		ExpandCommandBufferL(sizeof(TWsCmdHeaderBase));
sl@0
   806
sl@0
   807
		// Externalize the clipping region data from position after the header
sl@0
   808
		RBufWriteStream bufWriteStream;
sl@0
   809
		bufWriteStream.Open(*CurrentDrawCommandBuffer(), CurrentCommandBufferWritePos() + sizeof(TWsCmdHeaderBase));
sl@0
   810
		CleanupClosePushL(bufWriteStream);
sl@0
   811
		TInt dataLen = iLastDrawGc->ExternalizeClippingRegionL(bufWriteStream);
sl@0
   812
sl@0
   813
		// Setup the clipping region data header
sl@0
   814
		CWsClient::iCurrentCommand.iOpcode = EWsStoreClippingRegion;
sl@0
   815
		CWsClient::iCurrentCommand.iCmdLength = REINTERPRET_CAST(TInt16&,dataLen);
sl@0
   816
sl@0
   817
		// Store command header for clipping region data at current write position
sl@0
   818
		CommandBufferWrite(&CWsClient::iCurrentCommand,sizeof(TWsCmdHeaderBase));
sl@0
   819
sl@0
   820
		// Update write position for command data
sl@0
   821
		iCurrentSegment->iCurrentCommandBufferWritePos += dataLen;
sl@0
   822
		CleanupStack::PopAndDestroy(&bufWriteStream);
sl@0
   823
		}
sl@0
   824
	else
sl@0
   825
		{		
sl@0
   826
		TUint16 opcode = CWsClient::iCurrentCommand.iOpcode;
sl@0
   827
		CWsClient::iCurrentCommand.iOpcode |= aOpcodeFlags;
sl@0
   828
sl@0
   829
		// ensure room in command buffer
sl@0
   830
		ExpandCommandBufferL(sizeof(TWsCmdHeaderBase) + CWsClient::iCurrentCommand.iCmdLength);
sl@0
   831
sl@0
   832
		// Store command header to current position
sl@0
   833
		CommandBufferWrite(&CWsClient::iCurrentCommand, sizeof(TWsCmdHeaderBase));
sl@0
   834
sl@0
   835
		// If there's command data (other than header), store it
sl@0
   836
		if (CWsClient::iCurrentCommand.iCmdLength > 0)
sl@0
   837
			{
sl@0
   838
			CommandBufferWrite(aCmdData, CWsClient::iCurrentCommand.iCmdLength);
sl@0
   839
			}
sl@0
   840
sl@0
   841
		CWsClient::iCurrentCommand.iOpcode = opcode;
sl@0
   842
		}
sl@0
   843
	}
sl@0
   844
sl@0
   845
sl@0
   846
/*------------------------------------------------------------------------------
sl@0
   847
  Description: Stores graphics context information into the command buffer
sl@0
   848
               from the current write position.
sl@0
   849
 -----------------------------------------------------------------------------*/
sl@0
   850
void CWsRedrawMsgWindow::StoreAllGcAttributesL(CWsGc* aGc)
sl@0
   851
	{
sl@0
   852
	// In order for the externalize below to work correctly from
sl@0
   853
	// a non-zero position we have to create the header placeholder
sl@0
   854
	ExpandCommandBufferL(sizeof(TWsCmdHeaderBase));
sl@0
   855
sl@0
   856
	// Externalise GC attribute data. We do this before writing the
sl@0
   857
	// header as we do not know the size of the data yet and it is
sl@0
   858
	// part of the header.
sl@0
   859
	TInt numOfBytesAdded = aGc->ExternalizeL(*CurrentDrawCommandBuffer(),
sl@0
   860
				CurrentCommandBufferWritePos() + sizeof(TWsCmdHeaderBase));
sl@0
   861
sl@0
   862
	// Setup the header
sl@0
   863
	TWsCmdHeaderBase cmdHeader;
sl@0
   864
	cmdHeader.iCmdLength = (TInt16) numOfBytesAdded;		// as calculated above
sl@0
   865
	cmdHeader.iOpcode = (TInt16) EWsStoreAllGcAttributes;
sl@0
   866
sl@0
   867
	// Store the header for the GC data into the space we created
sl@0
   868
	CommandBufferWrite(&cmdHeader, sizeof(TWsCmdHeaderBase));
sl@0
   869
sl@0
   870
	// Update write position for command data
sl@0
   871
	iCurrentSegment->iCurrentCommandBufferWritePos += numOfBytesAdded;
sl@0
   872
	}
sl@0
   873
sl@0
   874
LOCAL_C void CallSegmentEnd(TAny* aAnnotationObserver)
sl@0
   875
	{
sl@0
   876
	static_cast<MWsDrawAnnotationObserver*>(aAnnotationObserver)->SegmentRedrawEnd();
sl@0
   877
	}
sl@0
   878
sl@0
   879
/*------------------------------------------------------------------------------
sl@0
   880
  Description: Loops through the whole of the current command buffer, processing
sl@0
   881
               each in turn.
sl@0
   882
 -----------------------------------------------------------------------------*/
sl@0
   883
void CWsRedrawMsgWindow::DrawCommandsL()
sl@0
   884
	{
sl@0
   885
	LOG_WINDOW_DRAW_COMMANDS_START(WsWin(), iRedrawRegion);
sl@0
   886
	WS_ASSERT_DEBUG(iMemoryLock > 0, EWsPanicMemoryLock);
sl@0
   887
	static TBuf8<EClientBufferMaxSize> buf;
sl@0
   888
	TInt regionCount = iRedrawSegments.Count();
sl@0
   889
	
sl@0
   890
	for (TInt regionNum = 0; regionNum < regionCount; ++regionNum)
sl@0
   891
		{
sl@0
   892
		CRedrawSegment* segment = iRedrawSegments[regionNum];
sl@0
   893
		LOG_REDRAW_SEGMENT(regionNum, segment->iRedrawSegmentType);
sl@0
   894
		if (segment->iRedrawSegmentType == ESegmentTypePendingRedraw)
sl@0
   895
			continue;
sl@0
   896
sl@0
   897
		// The amount of commands we process is given by the value of the
sl@0
   898
		// current write position rather than the size of the command buffer.
sl@0
   899
		// Note: the write position is incremented as each command is stored and
sl@0
   900
		// will typically be less than the buffer size.
sl@0
   901
		const TInt length = segment->iCurrentCommandBufferWritePos;
sl@0
   902
sl@0
   903
		// need to draw this region?
sl@0
   904
		const TRegion * localDrawRegion = 0;
sl@0
   905
		if (length)
sl@0
   906
		 	localDrawRegion = ReadRegion(regionNum);
sl@0
   907
		if (localDrawRegion)
sl@0
   908
			{
sl@0
   909
			MWsDrawAnnotationObserver* const annoObs = Screen()->DrawAnnotationObserver();
sl@0
   910
			if(annoObs)
sl@0
   911
				{
sl@0
   912
				annoObs->SegmentRedrawStart(*localDrawRegion);
sl@0
   913
				CleanupStack::PushL(TCleanupItem(CallSegmentEnd, annoObs));
sl@0
   914
				}
sl@0
   915
			CPlaybackGc::Instance()->SetTargetRegion(localDrawRegion);
sl@0
   916
			LOG_REDRAW_SEGMENT_REGION(localDrawRegion)
sl@0
   917
			
sl@0
   918
			TWsCmdHeaderBase header;
sl@0
   919
			TInt pos = 0; // Set to first command position in buffer
sl@0
   920
			CBufSeg* drawCmdBuffer = segment->iDrawCommands;
sl@0
   921
sl@0
   922
#ifdef _DEBUG
sl@0
   923
			// Read the first command header. The associated opcode must always be
sl@0
   924
			// EWsStoreAllGcAttributes as this is always the first stored item.
sl@0
   925
			drawCmdBuffer->Read(pos,&header,sizeof(TWsCmdHeaderBase));
sl@0
   926
			WS_ASSERT_DEBUG(header.iOpcode == EWsStoreAllGcAttributes, EWsPanicDrawCommandsBufferCorrupt);
sl@0
   927
#endif
sl@0
   928
sl@0
   929
			// Read through remaining commands
sl@0
   930
			while (pos < length)
sl@0
   931
				{
sl@0
   932
				// Get header of command
sl@0
   933
				drawCmdBuffer->Read(pos, &header, sizeof(TWsCmdHeaderBase));
sl@0
   934
				pos += sizeof(TWsCmdHeaderBase);
sl@0
   935
sl@0
   936
				switch(header.iOpcode)
sl@0
   937
					{
sl@0
   938
					case EWsStoreAllGcAttributes:
sl@0
   939
						{
sl@0
   940
						// Header indicates command encapsulates gc data
sl@0
   941
						CPlaybackGc::Instance()->Reset();
sl@0
   942
sl@0
   943
						// Read gc data
sl@0
   944
						CPlaybackGc::Instance()->InternalizeL(*drawCmdBuffer,pos);
sl@0
   945
sl@0
   946
						}
sl@0
   947
						break;
sl@0
   948
					case EWsStoreClippingRegion:
sl@0
   949
						{
sl@0
   950
						// Clipping region data read in from current position via stream
sl@0
   951
						RBufReadStream bufReadStream;
sl@0
   952
						bufReadStream.Open(*drawCmdBuffer,pos);
sl@0
   953
						CleanupClosePushL(bufReadStream);
sl@0
   954
						CPlaybackGc::Instance()->InternalizeClippingRegionL(bufReadStream);
sl@0
   955
						CleanupStack::PopAndDestroy(&bufReadStream);
sl@0
   956
						}
sl@0
   957
						break;
sl@0
   958
					default:
sl@0
   959
						{
sl@0
   960
						// Another type of command. Read it.
sl@0
   961
						CWsClient::iCurrentCommand.iCmdLength = header.iCmdLength;
sl@0
   962
						drawCmdBuffer->Read(pos,buf,header.iCmdLength);
sl@0
   963
sl@0
   964
						TInt opcode = header.iOpcode;
sl@0
   965
sl@0
   966
						// Drawing command?
sl@0
   967
						if (opcode & EWsGcOpFlagDrawOp)
sl@0
   968
							{
sl@0
   969
							opcode &= ~EWsGcOpFlagDrawOp;
sl@0
   970
							}
sl@0
   971
						if (opcode > -1)
sl@0
   972
							{
sl@0
   973
							LOG_PLAYBACK_GC_COMMAND(opcode, buf.Ptr())
sl@0
   974
							CPlaybackGc::Instance()->CommandL(static_cast<TWsGcOpcodes>(opcode),buf);
sl@0
   975
							}
sl@0
   976
						}
sl@0
   977
						break;
sl@0
   978
					}
sl@0
   979
				pos += header.iCmdLength; // Move on, header indicates length
sl@0
   980
				}
sl@0
   981
			if (annoObs)
sl@0
   982
				CleanupStack::PopAndDestroy(annoObs);
sl@0
   983
			}
sl@0
   984
		}
sl@0
   985
	LOG_WINDOW_DRAW_COMMANDS_END(WsWin());
sl@0
   986
	}
sl@0
   987
sl@0
   988
/*------------------------------------------------------------------------------
sl@0
   989
  Description: Called when the currently stored graphics commands
sl@0
   990
               are no longer required.
sl@0
   991
 -----------------------------------------------------------------------------*/
sl@0
   992
void CWsRedrawMsgWindow::DiscardStoredCommands()
sl@0
   993
	{
sl@0
   994
	iCurrentSegment = NULL;
sl@0
   995
	if (iRedrawSegments.Count() > 0)
sl@0
   996
		{
sl@0
   997
		// First of all, if we have any redraws pending, update the screen with
sl@0
   998
		// whatever commands we have before we throw them away:
sl@0
   999
		if (iFlags & EPendingScheduledDraw)
sl@0
  1000
			Screen()->DoRedrawNow();
sl@0
  1001
sl@0
  1002
		// for all regions or just Partial Redraw regions > index 0: delete bitmaps and draw commands
sl@0
  1003
		iRedrawSegments.ResetAndDestroy();
sl@0
  1004
sl@0
  1005
		iLastDrawGc = NULL;
sl@0
  1006
		}
sl@0
  1007
	}
sl@0
  1008
sl@0
  1009
void CWsRedrawMsgWindow::CreateNewSegmentL(const TRect& aRect, TRedrawSegmentType aNewRedrawRegionType)
sl@0
  1010
	{
sl@0
  1011
	CWsRedrawMsgWindow::CRedrawSegment* newRegion = CWsRedrawMsgWindow::CRedrawSegment::NewLC(aRect, aNewRedrawRegionType, *this);
sl@0
  1012
sl@0
  1013
	iRedrawSegments.AppendL(newRegion);
sl@0
  1014
	iCurrentSegment = newRegion;
sl@0
  1015
	CleanupStack::Pop(newRegion);
sl@0
  1016
sl@0
  1017
	// Set iLastDrawGc to NULL. This will cause all GC attributes to be stored
sl@0
  1018
	// in redraw store when the window receives the next command
sl@0
  1019
	iLastDrawGc = NULL;
sl@0
  1020
	}
sl@0
  1021
sl@0
  1022
void CWsRedrawMsgWindow::AddFbsBitmapsL(TInt aHandle, TInt aMaskHandle)
sl@0
  1023
	{
sl@0
  1024
	AddFbsBitmapRefL(aHandle);
sl@0
  1025
	if (aMaskHandle)
sl@0
  1026
		{
sl@0
  1027
		AddFbsBitmapRefL(aMaskHandle);
sl@0
  1028
		}
sl@0
  1029
	}
sl@0
  1030
sl@0
  1031
void CWsRedrawMsgWindow::AddFbsBitmapRefL(TInt aHandle)
sl@0
  1032
	{
sl@0
  1033
	TInt indexBitmapRef = iFbsBitmapRefArray.FindInOrder(aHandle, &FindBitmapRefByHandle);
sl@0
  1034
	
sl@0
  1035
	// check whether the window already has a reference for the bitmap through a segment other than the current one
sl@0
  1036
	if (indexBitmapRef >=0)
sl@0
  1037
	    {
sl@0
  1038
	    TBool handleAppend = iCurrentSegment->AppendNonDuplicateBitmapHandleL(aHandle);
sl@0
  1039
	    if (handleAppend)
sl@0
  1040
	        {
sl@0
  1041
	        iFbsBitmapRefArray[indexBitmapRef]->IncRefCount();
sl@0
  1042
	        }
sl@0
  1043
        }
sl@0
  1044
    else
sl@0
  1045
        {
sl@0
  1046
        CFbsBitmapRef* bitmapRef = new(ELeave) CFbsBitmapRef;
sl@0
  1047
        CleanupStack::PushL(bitmapRef);
sl@0
  1048
        if (bitmapRef->Duplicate(aHandle)!=KErrNone)
sl@0
  1049
            OwnerPanic(EWservPanicBitmap);
sl@0
  1050
        iFbsBitmapRefArray.InsertInOrderL(bitmapRef, TLinearOrder<CFbsBitmapRef>(InsertBitmapRefByHandle));
sl@0
  1051
        CleanupStack::Pop(bitmapRef);
sl@0
  1052
        bitmapRef->IncRefCount();
sl@0
  1053
#ifdef _DEBUG
sl@0
  1054
        TBool bitmapHandleAppended = EFalse;
sl@0
  1055
        TRAPD(err, bitmapHandleAppended = iCurrentSegment->AppendNonDuplicateBitmapHandleL(aHandle);
sl@0
  1056
                   WS_ASSERT_DEBUG(bitmapHandleAppended,EWsPanicUnexpectedBitmapHandleInArray););
sl@0
  1057
#else
sl@0
  1058
        TRAPD(err, iCurrentSegment->AppendNonDuplicateBitmapHandleL(aHandle););
sl@0
  1059
#endif
sl@0
  1060
        if (err != KErrNone)
sl@0
  1061
            {
sl@0
  1062
            // Cleanup the array, then propagate the error
sl@0
  1063
            indexBitmapRef = iFbsBitmapRefArray.FindInOrder(aHandle, &FindBitmapRefByHandle);
sl@0
  1064
            if (indexBitmapRef >= 0)
sl@0
  1065
                {
sl@0
  1066
                iFbsBitmapRefArray[indexBitmapRef]->DecRefCount();
sl@0
  1067
                delete iFbsBitmapRefArray[indexBitmapRef];
sl@0
  1068
                iFbsBitmapRefArray.Remove(indexBitmapRef);
sl@0
  1069
                }
sl@0
  1070
            User::Leave(err);
sl@0
  1071
            }
sl@0
  1072
        }
sl@0
  1073
	}
sl@0
  1074
sl@0
  1075
// Only called during playback of a redraw store segment
sl@0
  1076
CFbsBitmap* CWsRedrawMsgWindow::BitmapFromHandle(TInt aHandle) const
sl@0
  1077
    {
sl@0
  1078
    const TInt index = iFbsBitmapRefArray.FindInOrder(aHandle, &FindBitmapRefByHandle);
sl@0
  1079
    if (index != KErrNotFound)
sl@0
  1080
        {
sl@0
  1081
        CFbsBitmap* bitmap = iFbsBitmapRefArray[index];
sl@0
  1082
        return bitmap;
sl@0
  1083
        }
sl@0
  1084
    return NULL;
sl@0
  1085
    }
sl@0
  1086
sl@0
  1087
sl@0
  1088
void CWsRedrawMsgWindow::AddWsBitmapsL(TInt aHandle, TInt aMaskHandle)
sl@0
  1089
	{
sl@0
  1090
	if (iWsWin->WsOwner() == NULL)
sl@0
  1091
		Panic(EWsPanicDrawCommandsInvalidState);
sl@0
  1092
	DWsBitmap * bmp = static_cast<DWsBitmap*>(iWsWin->WsOwner()->HandleToObj(aHandle, WS_HANDLE_BITMAP));
sl@0
  1093
	if (!bmp)
sl@0
  1094
		OwnerPanic(EWservPanicBitmap);
sl@0
  1095
	iCurrentSegment->AddWsBitmapL(bmp);
sl@0
  1096
	if (aMaskHandle)
sl@0
  1097
		{
sl@0
  1098
		bmp = static_cast<DWsBitmap*>(iWsWin->WsOwner()->HandleToObj(aMaskHandle, WS_HANDLE_BITMAP));
sl@0
  1099
		if (!bmp)
sl@0
  1100
			OwnerPanic(EWservPanicBitmap);
sl@0
  1101
		iCurrentSegment->AddWsBitmapL(bmp);
sl@0
  1102
		}
sl@0
  1103
	}
sl@0
  1104
sl@0
  1105
void CWsRedrawMsgWindow::CRedrawSegment::AddWsBitmapL(DWsBitmap* bitmap)
sl@0
  1106
	{
sl@0
  1107
	iWsBitmapArray.AppendL(bitmap);
sl@0
  1108
	bitmap->IncRefCount();
sl@0
  1109
	}
sl@0
  1110
sl@0
  1111
void CWsRedrawMsgWindow::AddWsFontL(TInt aHandle)
sl@0
  1112
	{
sl@0
  1113
	if (iWsWin->WsOwner()==NULL)
sl@0
  1114
		Panic(EWsPanicDrawCommandsInvalidState);
sl@0
  1115
	TDblQueIter<CWsFbsFont> iter(CWsFontCache::List());
sl@0
  1116
	CWsFbsFont* font=NULL;
sl@0
  1117
	while((font=iter++)!=NULL)
sl@0
  1118
		{
sl@0
  1119
		if (font->Handle()==aHandle)
sl@0
  1120
			break;
sl@0
  1121
		}
sl@0
  1122
	if (font)
sl@0
  1123
		{
sl@0
  1124
		iCurrentSegment->iWsFontArray.AppendL(font);
sl@0
  1125
		++(font->iCount);
sl@0
  1126
		}
sl@0
  1127
	}
sl@0
  1128
sl@0
  1129
void CWsRedrawMsgWindow::CRedrawSegment::AddDrawerL(TGraphicDrawerId aDrawerId)
sl@0
  1130
	{
sl@0
  1131
	TInt error = iDrawerArray.InsertInOrder(aDrawerId, TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare));
sl@0
  1132
	if (error != KErrAlreadyExists && error != KErrNone)
sl@0
  1133
		{
sl@0
  1134
		User::Leave(error);
sl@0
  1135
		}
sl@0
  1136
	}
sl@0
  1137
sl@0
  1138
TBool CWsRedrawMsgWindow::CRedrawSegment::ContainsDrawers(const TArray<TGraphicDrawerId>& aDrawers,const TRegion& aRegion) const
sl@0
  1139
	{
sl@0
  1140
	TBool result = EFalse;
sl@0
  1141
	if (iDrawerArray.Count() > 0)
sl@0
  1142
		{
sl@0
  1143
		STACK_REGION tempRegion;
sl@0
  1144
		tempRegion.Intersection(iRegion, aRegion);
sl@0
  1145
		if (tempRegion.CheckError() || (tempRegion.Count() > 0) )
sl@0
  1146
			{ // regions do intersect, (presumed if region had an error); so check for a matching Id
sl@0
  1147
			const TInt drawersCount = aDrawers.Count();
sl@0
  1148
			for (TInt idx = 0; idx < drawersCount; ++idx)
sl@0
  1149
				{ // (iDrawerArray is kept sorted)
sl@0
  1150
				if (KErrNotFound != iDrawerArray.FindInOrder(aDrawers[idx], TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare)))
sl@0
  1151
					{
sl@0
  1152
					result = ETrue;
sl@0
  1153
					break;
sl@0
  1154
					}
sl@0
  1155
 					
sl@0
  1156
				const TInt count = iDrawerArray.Count();
sl@0
  1157
				for(TInt i = 0; i < count; i++)
sl@0
  1158
					{
sl@0
  1159
					const CWsGraphicDrawer* drawer = CWsTop::WindowServer()->ResolveGraphic(iDrawerArray[i]);
sl@0
  1160
					if(drawer && drawer->Contains(aDrawers))
sl@0
  1161
						{
sl@0
  1162
						result = ETrue;
sl@0
  1163
						break;
sl@0
  1164
						}
sl@0
  1165
					}
sl@0
  1166
				}
sl@0
  1167
			}
sl@0
  1168
		tempRegion.Close();
sl@0
  1169
		}
sl@0
  1170
	return result;
sl@0
  1171
	}
sl@0
  1172
sl@0
  1173
void CWsRedrawMsgWindow::AddWsDrawableSourceL(TInt aHandle)
sl@0
  1174
	{
sl@0
  1175
	if (iWsWin->WsOwner() == NULL)
sl@0
  1176
		Panic(EWsPanicDrawCommandsInvalidState);
sl@0
  1177
	CWsDrawableSource* drawableSource = static_cast<CWsDrawableSource*>(iWsWin->WsOwner()->HandleToObj(aHandle, WS_HANDLE_DRAWABLE_SOURCE));
sl@0
  1178
	if (!drawableSource)
sl@0
  1179
		OwnerPanic(EWservPanicDrawableSource);
sl@0
  1180
	iCurrentSegment->AddWsDrawableSourceL(drawableSource);
sl@0
  1181
	}
sl@0
  1182
sl@0
  1183
void CWsRedrawMsgWindow::CRedrawSegment::AddWsDrawableSourceL(CWsDrawableSource* aDrawableSource)
sl@0
  1184
	{
sl@0
  1185
	iWsDrawableSourceArray.AppendL(aDrawableSource);
sl@0
  1186
	aDrawableSource->IncRefCount();
sl@0
  1187
	}
sl@0
  1188
sl@0
  1189
inline TBool CWsRedrawMsgWindow::NoBuffer() const
sl@0
  1190
	{
sl@0
  1191
	return (iRedrawSegments.Count() == 0);
sl@0
  1192
	}
sl@0
  1193
sl@0
  1194
void CWsRedrawMsgWindow::ClientExposing()
sl@0
  1195
	{
sl@0
  1196
	Invalidate();
sl@0
  1197
	}
sl@0
  1198
sl@0
  1199
/*------------------------------------------------------------------------------
sl@0
  1200
  Description: If a complete set of drawing commands have been stored
sl@0
  1201
               this method attempts to draw ALL the commands via DrawCommandsL().
sl@0
  1202
               It also draws the window in the background colour if the window is
sl@0
  1203
               opaque.
sl@0
  1204
 -----------------------------------------------------------------------------*/
sl@0
  1205
void CWsRedrawMsgWindow::DrawWindow()
sl@0
  1206
	{
sl@0
  1207
	iFlags &= ~EPendingScheduledDraw;
sl@0
  1208
	// This is a happy window - it can draw itself whenever we ask.
sl@0
  1209
	CScreen* screen = Screen();
sl@0
  1210
	if ((screen->AutoClear() && (iFlags & EBackgroundClear)) || HasElement())
sl@0
  1211
		{
sl@0
  1212
		DrawBackgroundColor(*iRedrawRegion,(iFlags&EBackgroundClear)!=0);
sl@0
  1213
		}
sl@0
  1214
	if (HasElement())
sl@0
  1215
		{
sl@0
  1216
		screen->WindowElements().UnassignPlacedElements(*iRedrawRegion,*CliWin(),CPlaybackGc::Instance()->GcDrawingCount());
sl@0
  1217
		}
sl@0
  1218
	// If valid commands have been stored, draw them.
sl@0
  1219
	if (iRedrawSegments.Count() > 0)
sl@0
  1220
		{
sl@0
  1221
		Lock();
sl@0
  1222
		TRAP_IGNORE(DrawCommandsL());
sl@0
  1223
		Unlock();
sl@0
  1224
		}
sl@0
  1225
	if (HasElement())	//HasElement won't be cleared following first call above. May get set...
sl@0
  1226
		{
sl@0
  1227
		TInt state=screen->WindowElements().CleanUpPlacedElements(*CliWin(),CPlaybackGc::Instance()->GcDrawingCount());
sl@0
  1228
		if (state&CWindowElement::EFastPath)
sl@0
  1229
			{
sl@0
  1230
            if (HasElement())
sl@0
  1231
                {
sl@0
  1232
                screen->ElementAdded();
sl@0
  1233
                }
sl@0
  1234
            else
sl@0
  1235
                {
sl@0
  1236
                screen->ElementRemoved();
sl@0
  1237
                }
sl@0
  1238
			}
sl@0
  1239
		}
sl@0
  1240
	}
sl@0
  1241
sl@0
  1242
void CWsRedrawMsgWindow::RemoveFromRedrawQueueIfEmpty()
sl@0
  1243
	{
sl@0
  1244
	if (iInvalid.Count()==0)
sl@0
  1245
		{
sl@0
  1246
		iInvalid.Clear();	// Ensures heap cell is freed, otherwise may be left as an empty cell
sl@0
  1247
		iWsWin->WsOwner()->RedrawQueue()->RemoveInvalid(this);
sl@0
  1248
		}
sl@0
  1249
	}
sl@0
  1250
sl@0
  1251
const TRegion& CWsRedrawMsgWindow::InvalidArea() const
sl@0
  1252
	{
sl@0
  1253
	return(iInvalid);
sl@0
  1254
	}
sl@0
  1255
sl@0
  1256
TBool CWsRedrawMsgWindow::NeedsRedraw() const
sl@0
  1257
// If iInvalid has an persistant error it will not be reported as needing a redraw,
sl@0
  1258
// this is needed as otherwise cases where validation of a window results
sl@0
  1259
// in iInvalid having an error will get into an endless cycle of redraws.
sl@0
  1260
// The down side of this is that sometimes a window will not be sent a redraw
sl@0
  1261
// message when it needs it, some things can't be perfect!
sl@0
  1262
//
sl@0
  1263
	{
sl@0
  1264
	if ((!iWsWin->IsVisible()) || iInvalid.IsEmpty())
sl@0
  1265
		return EFalse;
sl@0
  1266
	
sl@0
  1267
	TRect nextRedrawRect;
sl@0
  1268
	return GetRedrawRect(nextRedrawRect);
sl@0
  1269
	}
sl@0
  1270
sl@0
  1271
TBool CWsRedrawMsgWindow::GetRedrawRect(TRect &aRect) const
sl@0
  1272
	{
sl@0
  1273
	if (iWsWin->ClientSetInvisible())
sl@0
  1274
		return EFalse;
sl@0
  1275
	
sl@0
  1276
	if(InRedraw())
sl@0
  1277
		{
sl@0
  1278
		aRect = iRedrawRect;
sl@0
  1279
		return (!aRect.IsEmpty());
sl@0
  1280
		}
sl@0
  1281
	else if(iInvalid.CheckError())
sl@0
  1282
		{
sl@0
  1283
		if (Screen()->ChangeTracking())
sl@0
  1284
			{
sl@0
  1285
			aRect = iWsWin->AbsRect();
sl@0
  1286
			}
sl@0
  1287
		else
sl@0
  1288
			{
sl@0
  1289
			if (iFlags & EStoringEntireWindow || iWsWin->VisibleRegion().CheckError())
sl@0
  1290
				{
sl@0
  1291
				aRect = iWsWin->AbsRect();
sl@0
  1292
				}
sl@0
  1293
			else
sl@0
  1294
				{
sl@0
  1295
				aRect = iWsWin->VisibleRegion().BoundingRect();
sl@0
  1296
				}
sl@0
  1297
			}
sl@0
  1298
		if (!(iFlags & EStoringEntireWindow))
sl@0
  1299
			iWsWin->ClipRectToViewport(aRect);
sl@0
  1300
		aRect.Move(-iWsWin->Origin());
sl@0
  1301
		return (!aRect.IsEmpty());
sl@0
  1302
		}
sl@0
  1303
	else if(iInvalid.Count())
sl@0
  1304
		{
sl@0
  1305
		if (iFlags & EStoringEntireWindow)
sl@0
  1306
			{
sl@0
  1307
			aRect = iInvalid.BoundingRect();
sl@0
  1308
			}
sl@0
  1309
		else
sl@0
  1310
			{
sl@0
  1311
			RWsRegion region;
sl@0
  1312
			region.Copy(iInvalid);
sl@0
  1313
			region.Offset(iWsWin->Origin());
sl@0
  1314
			if (!Screen()->ChangeTracking())
sl@0
  1315
				{
sl@0
  1316
				region.Intersect(iWsWin->VisibleRegion());
sl@0
  1317
				}
sl@0
  1318
			if (region.CheckError())
sl@0
  1319
				{
sl@0
  1320
				aRect = iInvalid.BoundingRect();
sl@0
  1321
				aRect.Move(iWsWin->Origin());
sl@0
  1322
				}
sl@0
  1323
			else
sl@0
  1324
				{
sl@0
  1325
				aRect = region.BoundingRect();
sl@0
  1326
				}
sl@0
  1327
			region.Close();
sl@0
  1328
			iWsWin->ClipRectToViewport(aRect);
sl@0
  1329
			aRect.Move(-iWsWin->Origin());
sl@0
  1330
			}
sl@0
  1331
		return (!aRect.IsEmpty());
sl@0
  1332
		}
sl@0
  1333
	else
sl@0
  1334
		{
sl@0
  1335
		return EFalse;
sl@0
  1336
		}
sl@0
  1337
	}
sl@0
  1338
sl@0
  1339
const TRegion &CWsRedrawMsgWindow::BaseDrawRegion() const
sl@0
  1340
	{
sl@0
  1341
	return(iWsWin->VisibleRegion());
sl@0
  1342
	}
sl@0
  1343
sl@0
  1344
void CWsRedrawMsgWindow::ClipInvalidRegion(const TRect &aRect)
sl@0
  1345
	{
sl@0
  1346
	if (iInvalid.Count()>0)
sl@0
  1347
		{
sl@0
  1348
		iInvalid.ClipRect(aRect);
sl@0
  1349
		RemoveFromRedrawQueueIfEmpty();
sl@0
  1350
		}
sl@0
  1351
	}
sl@0
  1352
sl@0
  1353
void CWsRedrawMsgWindow::EndRedraw()
sl@0
  1354
	{
sl@0
  1355
	if(!InRedraw())
sl@0
  1356
		OwnerPanic(EWservPanicDrawCommandsInvalidState);
sl@0
  1357
	if (iCurrentSegment)
sl@0
  1358
		{
sl@0
  1359
		iCurrentSegment->iDrawCommands->Compress();
sl@0
  1360
		if (iAtomic)
sl@0
  1361
			PromotePendingSegment();
sl@0
  1362
sl@0
  1363
		// Schedule an update of the area of the screen we just drew to:
sl@0
  1364
		iFlags |= EPendingScheduledDraw;
sl@0
  1365
		if (Screen()->ChangeTracking())
sl@0
  1366
			{
sl@0
  1367
			iWsWin->AddDirtyWindowRegion(iCurrentSegment->iRegion); // stored in window coordinates
sl@0
  1368
			if (iWsWin->IsVisible())
sl@0
  1369
				{
sl@0
  1370
				// Window is visible, (we're ignoring whether it's obscured or not)
sl@0
  1371
				// we need to send the new draw commands to the render stage.
sl@0
  1372
				Screen()->ScheduleWindow(iWsWin);
sl@0
  1373
				}
sl@0
  1374
			}
sl@0
  1375
		if (!iWsWin->HasBeenDrawnToScreen())
sl@0
  1376
			{
sl@0
  1377
			CliWin()->ScheduleRegionUpdate(NULL);
sl@0
  1378
			}
sl@0
  1379
		else if(!Screen()->ChangeTracking() && (iWsWin->VisibleRegion().Count() || iWsWin->VisibleRegion().CheckError())) // on screen? good.
sl@0
  1380
			{
sl@0
  1381
			STACK_REGION draw; //### in low memory where VisibleRegion() is intact we can degrade much better than this!
sl@0
  1382
			draw.Copy(iCurrentSegment->iRegion);
sl@0
  1383
			draw.Offset(iWsWin->Origin());
sl@0
  1384
			draw.Intersect(iWsWin->VisibleRegion());
sl@0
  1385
			if(!draw.CheckError())
sl@0
  1386
				Screen()->AddRedrawRegion(draw);
sl@0
  1387
			else
sl@0
  1388
				Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
sl@0
  1389
			draw.Close();
sl@0
  1390
			}
sl@0
  1391
		}
sl@0
  1392
sl@0
  1393
	iCurrentSegment = NULL;
sl@0
  1394
	
sl@0
  1395
	//store the result of the current redraw 
sl@0
  1396
	if(iFlags&ECurrentRedrawFailedStorage)
sl@0
  1397
		iFlags |= EPreviousRedrawFailedStorage; //set
sl@0
  1398
	else
sl@0
  1399
		iFlags &= ~EPreviousRedrawFailedStorage; //unset
sl@0
  1400
	iFlags &= ~ECurrentRedrawFailedStorage; //unset the flag for the next redraw
sl@0
  1401
	//
sl@0
  1402
sl@0
  1403
	iFlags&=~EBeginEndRedraw;
sl@0
  1404
	}
sl@0
  1405
sl@0
  1406
void CWsRedrawMsgWindow::ValidateRect(const TRect *aRect)
sl@0
  1407
	{
sl@0
  1408
	if (!WsWin()->BaseParent())
sl@0
  1409
		OwnerPanic(EWservPanicParentDeleted);
sl@0
  1410
	if (aRect)
sl@0
  1411
		iRedrawRect = *aRect;
sl@0
  1412
	if (!iInvalid.IsEmpty())
sl@0
  1413
		{
sl@0
  1414
		STACK_REGION validated;
sl@0
  1415
		validated.Copy(iInvalid);
sl@0
  1416
		if (aRect)
sl@0
  1417
			validated.ClipRect(iRedrawRect);
sl@0
  1418
		
sl@0
  1419
		if (iInvalid.CheckError())
sl@0
  1420
			{
sl@0
  1421
			iInvalid.Copy(iWsWin->VisibleRegion());
sl@0
  1422
			iInvalid.Offset(-iWsWin->Origin());
sl@0
  1423
			}
sl@0
  1424
		iInvalid.SubRegion(validated);
sl@0
  1425
		validated.Close();		
sl@0
  1426
		}
sl@0
  1427
	RemoveFromRedrawQueueIfEmpty();
sl@0
  1428
	}
sl@0
  1429
sl@0
  1430
TRgb CWsRedrawMsgWindow::BackColor() const
sl@0
  1431
	{
sl@0
  1432
	return(iBackColor);
sl@0
  1433
	}
sl@0
  1434
sl@0
  1435
/**
sl@0
  1436
This function used to be quite clever about what it invalidated and what it redrew by copying
sl@0
  1437
rectangles of the screen around.  This is a lot less subtle, and makes calling Scroll pretty much
sl@0
  1438
pointless, but it IS functionally correct.
sl@0
  1439
*/
sl@0
  1440
void CWsRedrawMsgWindow::Scroll(const TRect &aClipRect, const TPoint &aOffset,const TRect &aRect)
sl@0
  1441
	{
sl@0
  1442
	TRect rect = aRect;
sl@0
  1443
	rect.Intersection(aClipRect);	
sl@0
  1444
	Invalidate(&rect);
sl@0
  1445
	rect = aRect;
sl@0
  1446
	rect.Move(aOffset);
sl@0
  1447
	rect.Intersection(aClipRect);
sl@0
  1448
	Invalidate(&rect);
sl@0
  1449
	}
sl@0
  1450
sl@0
  1451
void CWsRedrawMsgWindow::ClearRedrawStore(TBool aClearPendingRedraw)
sl@0
  1452
	{
sl@0
  1453
	if(aClearPendingRedraw && (iFlags & EPendingScheduledDraw))
sl@0
  1454
		iFlags &= ~EPendingScheduledDraw;
sl@0
  1455
sl@0
  1456
	DiscardStoredCommands();
sl@0
  1457
	Invalidate();
sl@0
  1458
	}
sl@0
  1459
sl@0
  1460
sl@0
  1461
void CWsRedrawMsgWindow::PrepareForResizeL(const TSize& aSize, TSize& /*aOldSize*/)
sl@0
  1462
	{
sl@0
  1463
	TBool anyIncreases(EFalse);
sl@0
  1464
	if (aSize.iWidth>iWsWin->Size().iWidth||aSize.iHeight>iWsWin->Size().iHeight)
sl@0
  1465
		{
sl@0
  1466
		anyIncreases = ETrue;
sl@0
  1467
		}
sl@0
  1468
sl@0
  1469
	TRect newWinRect(TPoint(0,0),aSize);
sl@0
  1470
	iInvalid.ClipRect(newWinRect);
sl@0
  1471
	if (anyIncreases)
sl@0
  1472
		{
sl@0
  1473
		// add new invalid region to iInvalid
sl@0
  1474
		iInvalid.AddRect(newWinRect);
sl@0
  1475
		QueueRedraw();
sl@0
  1476
		iWsWin->WsOwner()->TriggerRedraw();
sl@0
  1477
		}
sl@0
  1478
	}
sl@0
  1479
sl@0
  1480
void CWsRedrawMsgWindow::Moved()
sl@0
  1481
	{
sl@0
  1482
	if (!(iFlags & EStoringEntireWindow))
sl@0
  1483
		{
sl@0
  1484
		DiscardSegmentsOutsideViewport();
sl@0
  1485
		}
sl@0
  1486
	if (iInvalid.Count())
sl@0
  1487
		{
sl@0
  1488
		QueueRedraw();
sl@0
  1489
		iWsWin->WsOwner()->TriggerRedraw();
sl@0
  1490
		}
sl@0
  1491
	}
sl@0
  1492
sl@0
  1493
TBool CWsRedrawMsgWindow::Contains(const TArray<TGraphicDrawerId>& aDrawers,const TRegion& aRegion) const
sl@0
  1494
	{
sl@0
  1495
	if (iRedrawSegments.Count() > 0)
sl@0
  1496
		{
sl@0
  1497
		// scan redraw store: calls Contains() on every region drawing commands are stored for,
sl@0
  1498
		// looking for a DrawWsGraphic command that intersects the aRegion
sl@0
  1499
		TBool contains = EFalse;
sl@0
  1500
		const TInt regionCount = iRedrawSegments.Count();
sl@0
  1501
		
sl@0
  1502
		// Apply an origin correction. The input aRegion is screen-absolute, while the redraw regions are window-relative.
sl@0
  1503
		STACK_REGION relRegion;
sl@0
  1504
		relRegion.Copy(aRegion);
sl@0
  1505
		relRegion.Offset(-(CliWin()->Origin().iX), -(CliWin()->Origin().iY));
sl@0
  1506
sl@0
  1507
		// loop through regions, stops when a match is found
sl@0
  1508
		for (TInt regionNum = 0; (regionNum < regionCount) && !contains; ++regionNum)
sl@0
  1509
			{
sl@0
  1510
			contains = iRedrawSegments[regionNum]->ContainsDrawers(aDrawers, relRegion);
sl@0
  1511
			}
sl@0
  1512
		relRegion.Close();
sl@0
  1513
		return contains;
sl@0
  1514
		}
sl@0
  1515
	else
sl@0
  1516
		{
sl@0
  1517
		return CWsWindowRedraw::Contains(aDrawers,aRegion);
sl@0
  1518
		}
sl@0
  1519
	}
sl@0
  1520
sl@0
  1521
TBool CWsRedrawMsgWindow::RedrawingInProgress() const
sl@0
  1522
	{
sl@0
  1523
	return (iFlags & EBeginEndRedraw);	
sl@0
  1524
	}
sl@0
  1525
sl@0
  1526
void CWsRedrawMsgWindow::WindowClosing()
sl@0
  1527
	{
sl@0
  1528
	iWsWin->WsOwner()->RedrawQueue()->RemoveInvalid(this);
sl@0
  1529
	CWsWindowRedraw::WindowClosing();
sl@0
  1530
	}
sl@0
  1531
sl@0
  1532
TBool CWsRedrawMsgWindow::IsRedrawStoreEmpty() const
sl@0
  1533
	{
sl@0
  1534
	return (iRedrawSegments.Count() <= 0); // Begin and End redraw are in the store too.
sl@0
  1535
	}
sl@0
  1536
sl@0
  1537
TBool CWsRedrawMsgWindow::IsBackgroundClearEnabled() const
sl@0
  1538
	{
sl@0
  1539
	return ((iFlags & EBackgroundClear) != 0);
sl@0
  1540
	}
sl@0
  1541
sl@0
  1542
void CWsRedrawMsgWindow::CleanupBitmapRefArray(const RArray<TInt>& aHandleArray)
sl@0
  1543
    {
sl@0
  1544
    TInt count = aHandleArray.Count();
sl@0
  1545
    for(TInt ii = count - 1 ; ii >= 0; ii--)
sl@0
  1546
        {
sl@0
  1547
        const TInt indexBitmapRef = iFbsBitmapRefArray.FindInOrder(aHandleArray[ii], &FindBitmapRefByHandle);
sl@0
  1548
        WS_ASSERT_DEBUG(indexBitmapRef >= 0,EWsPanicUnexpectedBitmapHandleInArray);
sl@0
  1549
        if (indexBitmapRef>=0)
sl@0
  1550
            {
sl@0
  1551
            iFbsBitmapRefArray[indexBitmapRef]->DecRefCount();
sl@0
  1552
            if (iFbsBitmapRefArray[indexBitmapRef]->RefCount()==0)
sl@0
  1553
                {
sl@0
  1554
                delete iFbsBitmapRefArray[indexBitmapRef];
sl@0
  1555
                iFbsBitmapRefArray.Remove(indexBitmapRef);
sl@0
  1556
                }
sl@0
  1557
            }
sl@0
  1558
        }
sl@0
  1559
    }
sl@0
  1560
sl@0
  1561
void CWsRedrawMsgWindow::SetScope(TScope aScope)
sl@0
  1562
	{
sl@0
  1563
	if (aScope == EStoreEntireWindow)
sl@0
  1564
		{
sl@0
  1565
		if (!(iFlags & EStoringEntireWindow))
sl@0
  1566
			{
sl@0
  1567
			iFlags |= EStoringEntireWindow;
sl@0
  1568
			Invalidate();
sl@0
  1569
			}
sl@0
  1570
		}
sl@0
  1571
	else
sl@0
  1572
		{
sl@0
  1573
		if (iFlags & EStoringEntireWindow)
sl@0
  1574
			{
sl@0
  1575
			iFlags &= ~ EStoringEntireWindow;
sl@0
  1576
			DiscardSegmentsOutsideViewport();
sl@0
  1577
			}
sl@0
  1578
		}
sl@0
  1579
	}
sl@0
  1580
sl@0
  1581
/**
sl@0
  1582
Removes all segments from the redraw store which are outside the viewport onto the window.
sl@0
  1583
Note that this doesn't clip the regions of those segments which are partly outside, since
sl@0
  1584
this wouldn't actually achieve anything useful.
sl@0
  1585
sl@0
  1586
This function allocates memory so it is not suitable to run as part of ReleaseMemory.
sl@0
  1587
*/
sl@0
  1588
TBool CWsRedrawMsgWindow::DiscardSegmentsOutsideViewport()
sl@0
  1589
	{
sl@0
  1590
	TBool discarded = EFalse;
sl@0
  1591
	TInt count = iRedrawSegments.Count();
sl@0
  1592
	STACK_REGION viewport;
sl@0
  1593
	CliWin()->GetClippedBaseArea(viewport);
sl@0
  1594
	viewport.Offset(-iWsWin->Origin());
sl@0
  1595
	STACK_REGION intersect;
sl@0
  1596
	for (TInt idx = count - 1; idx >= 0; --idx)
sl@0
  1597
		{
sl@0
  1598
		CRedrawSegment * segment = iRedrawSegments[idx];
sl@0
  1599
		intersect.Intersection(segment->iRegion, viewport);
sl@0
  1600
		if (!intersect.CheckError() && intersect.IsEmpty())
sl@0
  1601
			{
sl@0
  1602
			iInvalid.Union(segment->iRegion);
sl@0
  1603
			delete segment;
sl@0
  1604
			iRedrawSegments.Remove(idx);
sl@0
  1605
			if (iCurrentSegment == segment)
sl@0
  1606
				iCurrentSegment = NULL;
sl@0
  1607
			discarded = ETrue;
sl@0
  1608
			}
sl@0
  1609
		}
sl@0
  1610
	intersect.Close();
sl@0
  1611
	viewport.Close();
sl@0
  1612
	return discarded;
sl@0
  1613
	}
sl@0
  1614
sl@0
  1615
/**
sl@0
  1616
Statements encapsulated in between Lock() and Unlock() is guaranteed to execute in an 
sl@0
  1617
atomic way without being interupted by a call to ReleaseMemory from CWsMemoryManager. 
sl@0
  1618
Locking will prevent memory belonging to this object to be freed during a 
sl@0
  1619
memory alloc/realloc originating from self.
sl@0
  1620
*/
sl@0
  1621
void CWsRedrawMsgWindow::Lock()
sl@0
  1622
	{
sl@0
  1623
	++iMemoryLock;
sl@0
  1624
	}
sl@0
  1625
	
sl@0
  1626
void CWsRedrawMsgWindow::Unlock()
sl@0
  1627
	{
sl@0
  1628
	--iMemoryLock;
sl@0
  1629
	WS_ASSERT_DEBUG(iMemoryLock >= 0, EWsPanicMemoryLock);
sl@0
  1630
	}
sl@0
  1631
sl@0
  1632
TBool CWsRedrawMsgWindow::ReleaseMemory(MWsMemoryRelease::TMemoryReleaseLevel aLevel)
sl@0
  1633
	{
sl@0
  1634
	//When this function is called, wserv is in the middle of executing something.
sl@0
  1635
	//Therefore we can not safely do anything that alters the state of any shared 
sl@0
  1636
	//resouces (like e.g. CScreenRedraw::iInvalid).
sl@0
  1637
	//In addition, we should refrain from anything that might try to allocate memory.
sl@0
  1638
	TBool released = EFalse;
sl@0
  1639
	//Don't release iRedrawSegments from this win if its currently being rendered, 
sl@0
  1640
	//is releasing memory or is receiving drawcommands.	
sl@0
  1641
	if (iMemoryLock == 0 && !iCurrentSegment)
sl@0
  1642
		{
sl@0
  1643
		Lock();
sl@0
  1644
		switch (aLevel)
sl@0
  1645
			{
sl@0
  1646
			case MWsMemoryRelease::ELow:
sl@0
  1647
				break;
sl@0
  1648
			case MWsMemoryRelease::EMedium:
sl@0
  1649
				break;
sl@0
  1650
			case MWsMemoryRelease::EHigh:
sl@0
  1651
				//Only release memory from background windows.
sl@0
  1652
				if (iRedrawSegments.Count() > 0 && iWsWin->VisibleRegion().IsEmpty())
sl@0
  1653
					{
sl@0
  1654
					ReleaseRedrawSegments();
sl@0
  1655
					released = ETrue;
sl@0
  1656
					}
sl@0
  1657
				break;
sl@0
  1658
			}
sl@0
  1659
		Unlock();
sl@0
  1660
		}
sl@0
  1661
	return released;
sl@0
  1662
	}
sl@0
  1663
sl@0
  1664
void CWsRedrawMsgWindow::ReleaseRedrawSegments()
sl@0
  1665
	{
sl@0
  1666
 	iLastDrawGc = NULL;
sl@0
  1667
 	iCurrentSegment = NULL;
sl@0
  1668
 	iRedrawSegments.ResetAndDestroy();
sl@0
  1669
 	
sl@0
  1670
 	//The call to ResetAndDestroy just freed some memory so it should be 
sl@0
  1671
 	//possible to call Invalidate() now.
sl@0
  1672
 	Invalidate(); 
sl@0
  1673
 	
sl@0
  1674
 	//Releasing the same window over and over again could quickly end up in 
sl@0
  1675
 	//a never ending loop with a high-prio client before we find the window
sl@0
  1676
 	//that has nicked all memory. So call accessed now to prevent that.
sl@0
  1677
  	iWsWin->Accessed(); 
sl@0
  1678
   	}
sl@0
  1679
sl@0
  1680
void CWsRedrawMsgWindow::VisibleRegionChange()
sl@0
  1681
	{
sl@0
  1682
	if (!iFlags & EStoringEntireWindow)
sl@0
  1683
		{
sl@0
  1684
		DiscardSegmentsOutsideViewport();
sl@0
  1685
		}
sl@0
  1686
	if ((!iInvalid.IsEmpty()) && (!iWsWin->VisibleRegion().IsEmpty()))
sl@0
  1687
		{
sl@0
  1688
		STACK_REGION exposed;
sl@0
  1689
		exposed.Copy(iInvalid);
sl@0
  1690
		exposed.Offset(iWsWin->Origin());
sl@0
  1691
		exposed.Intersect(iWsWin->VisibleRegion());
sl@0
  1692
		if (!exposed.IsEmpty())
sl@0
  1693
			{
sl@0
  1694
			QueueRedraw();
sl@0
  1695
			}
sl@0
  1696
		exposed.Close();
sl@0
  1697
		}
sl@0
  1698
	}
sl@0
  1699
sl@0
  1700
TBool CWsRedrawMsgWindow::ReadyToDraw() const
sl@0
  1701
	{
sl@0
  1702
	//We are only ready to draw when we have a complete segment.
sl@0
  1703
	if (iWsWin->HasBeenDrawnToScreen())
sl@0
  1704
		return ETrue;
sl@0
  1705
	
sl@0
  1706
	if (iRedrawSegments.Count() == 0)
sl@0
  1707
		return EFalse;
sl@0
  1708
	
sl@0
  1709
	if (iRedrawSegments.Count() > 1)
sl@0
  1710
		return ETrue;
sl@0
  1711
	
sl@0
  1712
	if (iRedrawSegments[0]->iRedrawSegmentType == ESegmentTypePendingRedraw)
sl@0
  1713
		return EFalse;
sl@0
  1714
	
sl@0
  1715
	return ETrue;
sl@0
  1716
	}
sl@0
  1717
sl@0
  1718
TInt CWsRedrawMsgWindow::SizeInBytes() const
sl@0
  1719
	{
sl@0
  1720
	TInt size = sizeof(CWsRedrawMsgWindow);
sl@0
  1721
	for(TInt i = iRedrawSegments.Count()-1; i >= 0; i--)
sl@0
  1722
		{
sl@0
  1723
		size += iRedrawSegments[i]->SizeInBytes();
sl@0
  1724
		}
sl@0
  1725
	size += iInvalid.Count() * sizeof(TRect);
sl@0
  1726
	size += iLocalRedrawRegion.Count() * sizeof(TRect);
sl@0
  1727
	return size;
sl@0
  1728
	}
sl@0
  1729
sl@0
  1730
void CWsRedrawMsgWindow::PromotePendingSegment()
sl@0
  1731
	{
sl@0
  1732
	if (iRedrawSegments.Count() > 0 && iRedrawSegments[iRedrawSegments.Count() - 1]->iRedrawSegmentType == ESegmentTypePendingRedraw)
sl@0
  1733
		{
sl@0
  1734
		CRedrawSegment * segment = iRedrawSegments[iRedrawSegments.Count() - 1];
sl@0
  1735
		const TRect * rect = segment->iRegion.RectangleList();
sl@0
  1736
		// when we get here there should only ever be one rectangle in the region, but we are playing safe
sl@0
  1737
		for (TInt r = 0; r < segment->iRegion.Count(); ++r)
sl@0
  1738
			{
sl@0
  1739
			SubtractRectFromSegmentArray(*rect);			
sl@0
  1740
			++rect;
sl@0
  1741
			}
sl@0
  1742
		segment->iRedrawSegmentType = ESegmentTypeRedraw;
sl@0
  1743
		}
sl@0
  1744
	}
sl@0
  1745
sl@0
  1746
CFbsBitmapRef::CFbsBitmapRef()
sl@0
  1747
    {
sl@0
  1748
    }
sl@0
  1749
sl@0
  1750
CFbsBitmapRef::~CFbsBitmapRef()
sl@0
  1751
    {
sl@0
  1752
    WS_ASSERT_DEBUG(iRefCount==0,EWsPanicCounterValue);
sl@0
  1753
    }