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