os/graphics/windowing/windowserver/nonnga/SERVER/ScreenRedraw.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) 2006-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
// SCREEN_REDRAW.CPP
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include "ScreenRedraw.h"
sl@0
    19
sl@0
    20
#include <hal.h>
sl@0
    21
sl@0
    22
#include "debugbar.h"
sl@0
    23
#include "screen.h"
sl@0
    24
#include "inifile.h"
sl@0
    25
#include "offscreenbitmap.h"
sl@0
    26
#include "wspluginmanager.h"
sl@0
    27
#include "pointer.h"
sl@0
    28
#include "rootwin.h"
sl@0
    29
#include "walkwindowtree.h"
sl@0
    30
#include "wstop.h"
sl@0
    31
#include "WsMemMgr.h"
sl@0
    32
#include "Graphics/WsRenderStageFactory.h"
sl@0
    33
#include "Graphics/WsRenderStage.h"
sl@0
    34
#include "EVENT.H"
sl@0
    35
sl@0
    36
GLREF_D CDebugLogBase *wsDebugLog;
sl@0
    37
sl@0
    38
#ifdef USE_DEBUG_REGIONS
sl@0
    39
#	define DEBUG_REGION(col,fill,reg) DebugRegion(col,fill,reg)
sl@0
    40
#	define DEBUG_RECT(col,fill,rect) DebugRect(col,fill,rect)
sl@0
    41
#else
sl@0
    42
#	define DEBUG_REGION(col,fill,reg)
sl@0
    43
#	define DEBUG_RECT(col,fill,rect)
sl@0
    44
#endif
sl@0
    45
sl@0
    46
#if defined(__WINS__) && defined(_DEBUG)
sl@0
    47
#	define DEBUGOSB { CWsOffScreenBitmap * ofb = iScreen.OffScreenBitmap(); if (ofb) ofb->Update(); }
sl@0
    48
#else
sl@0
    49
#	define DEBUGOSB
sl@0
    50
#endif				
sl@0
    51
sl@0
    52
#ifdef _DEBUG
sl@0
    53
# define LOG_SCREEN_REDRAW_START {if (wsDebugLog) {_LIT(KLogScreenRedrawStart, ">> CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawStart);}}
sl@0
    54
# define LOG_SCREEN_REDRAW_END {if (wsDebugLog) {_LIT(KLogScreenRedrawEnd, "<< CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawEnd);}}
sl@0
    55
#else
sl@0
    56
# define LOG_SCREEN_REDRAW_START
sl@0
    57
# define LOG_SCREEN_REDRAW_END
sl@0
    58
#endif
sl@0
    59
sl@0
    60
CScreenRedraw::TTimedRect::TTimedRect(const TRect& aRect, const TTime& aTime):
sl@0
    61
	iRect(aRect), iTime(aTime)
sl@0
    62
	{
sl@0
    63
	}
sl@0
    64
sl@0
    65
TInt CScreenRedraw::TTimedRect::Compare(const TTimedRect& aOne,const TTimedRect& aOther)
sl@0
    66
	{
sl@0
    67
	if(aOne.iTime < aOther.iTime)
sl@0
    68
		return -1;
sl@0
    69
	else if(aOne.iTime > aOther.iTime)
sl@0
    70
		return 1;
sl@0
    71
	else
sl@0
    72
		return 0;
sl@0
    73
	}
sl@0
    74
sl@0
    75
CScreenRedraw * CScreenRedraw::NewL(CScreen& aScreen)
sl@0
    76
	{
sl@0
    77
	CScreenRedraw * self = new (ELeave) CScreenRedraw(aScreen);
sl@0
    78
	CleanupStack::PushL(self);
sl@0
    79
	self->ConstructL();
sl@0
    80
	CleanupStack::Pop(self);
sl@0
    81
	return self;
sl@0
    82
	}
sl@0
    83
	
sl@0
    84
CScreenRedraw::CScreenRedraw(CScreen& aScreen): iScreen(aScreen)
sl@0
    85
	{
sl@0
    86
	}
sl@0
    87
sl@0
    88
CScreenRedraw::~CScreenRedraw()
sl@0
    89
	{
sl@0
    90
	CWsRenderStage * stage = iRenderStages;
sl@0
    91
	while (stage!=NULL)
sl@0
    92
   		{
sl@0
    93
		CWsRenderStage *next=stage->Next();
sl@0
    94
   		delete stage;
sl@0
    95
		stage=next;
sl@0
    96
   		}
sl@0
    97
	iTimedDrawRect.Close();
sl@0
    98
	iInvalid.Close();
sl@0
    99
	iTopLayer.Close();
sl@0
   100
	iBannedRegion.Close();	
sl@0
   101
	}
sl@0
   102
sl@0
   103
void CScreenRedraw::ConstructL()
sl@0
   104
	{
sl@0
   105
	CWsPluginManager * pluginMgr = CWsTop::WindowServer()->PluginManager();
sl@0
   106
	
sl@0
   107
	// Setup the render stages for this screen:
sl@0
   108
	_LIT(KDefaultRenderStages, "std");
sl@0
   109
	_LIT(KDefaultFlickerFreeRenderStages, "flickerbuffer std");
sl@0
   110
	_LIT(KRenderStages,"RENDERSTAGES");
sl@0
   111
	TPtrC stagesString;
sl@0
   112
	const TBool customStages = WsIniFile->FindVar(iScreen.ScreenNumber(),KRenderStages,stagesString);
sl@0
   113
	
sl@0
   114
	// If noone specifies stages for this screen, assume the standard implementation:
sl@0
   115
	const TDesC * stages;
sl@0
   116
	if (customStages)
sl@0
   117
		stages = &stagesString;
sl@0
   118
	else if (iScreen.OffScreenBitmap())
sl@0
   119
		stages = &KDefaultFlickerFreeRenderStages();
sl@0
   120
	else
sl@0
   121
		stages = &KDefaultRenderStages();
sl@0
   122
	
sl@0
   123
	CWsRenderStage * lastStage = 0;
sl@0
   124
sl@0
   125
	// Parse the string for implementation IDs:
sl@0
   126
	TLex lex(*stages);
sl@0
   127
	while(true)
sl@0
   128
		{
sl@0
   129
		TPtrC ptr = lex.NextToken();
sl@0
   130
		if (ptr.Length() > 0)
sl@0
   131
			{
sl@0
   132
			TInt err = KErrNone;
sl@0
   133
			MWsRenderStageFactory * factory = pluginMgr->FindNamedImplementation<MWsRenderStageFactory>(ptr);
sl@0
   134
			if (factory)
sl@0
   135
				{
sl@0
   136
				CWsRenderStage * stage = 0;
sl@0
   137
				TRAP(err, stage = factory->CreateStageL(static_cast<MWsScreen*>(&iScreen), this));
sl@0
   138
				if (err == KErrNone)
sl@0
   139
					{
sl@0
   140
					if (!stage)
sl@0
   141
						{
sl@0
   142
						err = KErrNotFound;
sl@0
   143
						}
sl@0
   144
					else
sl@0
   145
						{
sl@0
   146
						if (lastStage)
sl@0
   147
							lastStage->SetNext(stage);
sl@0
   148
						else
sl@0
   149
							iRenderStages = stage;
sl@0
   150
						lastStage = stage;
sl@0
   151
						}
sl@0
   152
					}
sl@0
   153
				}
sl@0
   154
			else
sl@0
   155
				{
sl@0
   156
				err = KErrNotFound;
sl@0
   157
				}
sl@0
   158
sl@0
   159
			if (wsDebugLog)
sl@0
   160
				{
sl@0
   161
				TBuf<64> buf;
sl@0
   162
				if (err == KErrNone)
sl@0
   163
					{
sl@0
   164
					_LIT(KAddedRenderStage,"Added render stage: ");
sl@0
   165
					buf.Append(KAddedRenderStage);
sl@0
   166
					buf.Append(ptr);
sl@0
   167
					wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,buf);
sl@0
   168
					}
sl@0
   169
				else
sl@0
   170
					{
sl@0
   171
					_LIT(KMissingRenderStage,"Failed to add render stage (%d): ");
sl@0
   172
					buf.Append(KMissingRenderStage);
sl@0
   173
					buf.Append(ptr);
sl@0
   174
					wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,buf,err);
sl@0
   175
					}
sl@0
   176
				}
sl@0
   177
			}
sl@0
   178
		else
sl@0
   179
			{
sl@0
   180
			break;
sl@0
   181
			}
sl@0
   182
		}
sl@0
   183
	}
sl@0
   184
sl@0
   185
const TTime& CScreenRedraw::Now() const
sl@0
   186
	{
sl@0
   187
	if(!iAnimating)
sl@0
   188
		{
sl@0
   189
		iNow.UniversalTime();
sl@0
   190
		}
sl@0
   191
	return iNow;
sl@0
   192
	}
sl@0
   193
sl@0
   194
void CScreenRedraw::ScheduleRender(const TTimeIntervalMicroSeconds& aFromNow)
sl@0
   195
	{
sl@0
   196
	iRenderScheduled = ETrue;
sl@0
   197
	TTime then(Now() + aFromNow);
sl@0
   198
	if ((!iScheduled) || then < iNext)
sl@0
   199
		iNext = then;
sl@0
   200
	iScheduled = ETrue;
sl@0
   201
	CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
sl@0
   202
	}
sl@0
   203
sl@0
   204
void CScreenRedraw::ScheduleRedraw()
sl@0
   205
	{
sl@0
   206
	iNext = Now();
sl@0
   207
	iScheduled = ETrue;
sl@0
   208
		
sl@0
   209
	// The other scheduler also removes future animations which this one encompasses.
sl@0
   210
	// We choose not to do the region calculations needed to achieve that here.
sl@0
   211
	MWsAnimationScheduler* animSched=CWsTop::WindowServer()->AnimationScheduler();
sl@0
   212
	if (animSched)
sl@0
   213
		{
sl@0
   214
		animSched->ScheduleRedraw(iScreen,iNext);
sl@0
   215
		}
sl@0
   216
	}
sl@0
   217
sl@0
   218
void CScreenRedraw::ScheduleAnimation(const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow,const TTimeIntervalMicroSeconds& /*aFreq*/,const TTimeIntervalMicroSeconds& /*aStop*/)
sl@0
   219
	{
sl@0
   220
	TRect test(aRect);
sl@0
   221
	test.Intersection(iScreen.DrawableArea());
sl@0
   222
	if(!test.IsEmpty())
sl@0
   223
		{
sl@0
   224
		const TTime then(Now() + aFromNow);
sl@0
   225
		TTimedRect tRect(aRect, then);
sl@0
   226
sl@0
   227
		const TInt error = iTimedDrawRect.InsertInOrderAllowRepeats(tRect,TTimedRect::Compare);
sl@0
   228
		if (KErrNone == error)
sl@0
   229
	   		{
sl@0
   230
			if (iScheduled)
sl@0
   231
				{
sl@0
   232
				if (then < iNext)
sl@0
   233
					{
sl@0
   234
					iNext = then;
sl@0
   235
					}
sl@0
   236
				}
sl@0
   237
			else
sl@0
   238
				{
sl@0
   239
				iNext = then;
sl@0
   240
				iScheduled = ETrue;
sl@0
   241
				}
sl@0
   242
			// remove further futures that are completely contained
sl@0
   243
			TInt count = iTimedDrawRect.Count();
sl@0
   244
			for(TInt i=0; i<count; i++)
sl@0
   245
				{
sl@0
   246
				const TTimedRect& future = iTimedDrawRect[i];
sl@0
   247
				if(future.iTime.Int64() > then.Int64())
sl@0
   248
					{
sl@0
   249
					TRect rect(aRect);
sl@0
   250
					rect.BoundingRect(future.iRect);
sl@0
   251
					if(rect == aRect) // future is completely contained within aRect
sl@0
   252
						{
sl@0
   253
						iTimedDrawRect.Remove(i);
sl@0
   254
						count--;
sl@0
   255
						i--;
sl@0
   256
						}
sl@0
   257
					}
sl@0
   258
				}
sl@0
   259
			CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
sl@0
   260
sl@0
   261
			// Blue rectangles for scheduled animations
sl@0
   262
			DEBUG_RECT(TRgb(0x00, 0x00, 0xFF),TRgb(0x00, 0x00, 0xFF, 0x20),&aRect);
sl@0
   263
			}
sl@0
   264
		}
sl@0
   265
	}
sl@0
   266
sl@0
   267
// This adds a region to the stored invalid region.
sl@0
   268
// The invalid region is the area of the screen that needs to be redrawn in addition to any animations.
sl@0
   269
// The draw region is the area of the screen on which only the top window needs to be redrawn.
sl@0
   270
// If the top window has transparency, this can only be true when it has been made newly visible.
sl@0
   271
// The value of aSchedule could be determined automatically from iAnimating, but passing it this way
sl@0
   272
// allows us to have the assert, which is a very valuable assert.
sl@0
   273
void CScreenRedraw::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
sl@0
   274
	{
sl@0
   275
	WS_ASSERT_DEBUG(!iAnimating || !aSchedule, EWsPanicScheduledRedraw);
sl@0
   276
	
sl@0
   277
	if(aRegion.CheckError())
sl@0
   278
		{
sl@0
   279
		iInvalid.ForceError();
sl@0
   280
sl@0
   281
		if (aSchedule)
sl@0
   282
			ScheduleRedraw();
sl@0
   283
		}
sl@0
   284
	else if(aRegion.Count()) // often called despite window not being visible
sl@0
   285
		{
sl@0
   286
		if (aDepth == ERedrawAll)
sl@0
   287
			{
sl@0
   288
			// red lines for an invalid region which is ready to be drawn
sl@0
   289
			DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x20),&aRegion);
sl@0
   290
				
sl@0
   291
			iInvalid.Union(aRegion);
sl@0
   292
			
sl@0
   293
			if (aSchedule)
sl@0
   294
				ScheduleRedraw();
sl@0
   295
			}
sl@0
   296
		else
sl@0
   297
			{
sl@0
   298
			// yellow lines for a valid region which we will draw on top of
sl@0
   299
			DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x20),&aRegion);
sl@0
   300
				
sl@0
   301
			iTopLayer.Union(aRegion);
sl@0
   302
			
sl@0
   303
			if (aSchedule)
sl@0
   304
				ScheduleRedraw();			
sl@0
   305
			}
sl@0
   306
		}
sl@0
   307
	}
sl@0
   308
sl@0
   309
// This causes any asynchronously scheduled redraw to happen immediately
sl@0
   310
// It should be avoided where possible for performance reasons, but is 
sl@0
   311
// needed whenever the redraw store is discarded for a window which still
sl@0
   312
// has a redraw region pending.
sl@0
   313
void CScreenRedraw::DoRedrawNow()
sl@0
   314
	{
sl@0
   315
	if(!iAnimating)
sl@0
   316
		CWsTop::WindowServer()->AnimationScheduler()->DoRedrawNow(iScreen);
sl@0
   317
	}
sl@0
   318
	
sl@0
   319
#ifdef USE_DEBUG_REGIONS
sl@0
   320
void CScreenRedraw::DebugRect(TRgb aColor, TRgb aFill, const TRect* aRect)
sl@0
   321
	{
sl@0
   322
	if (aRect)
sl@0
   323
		{
sl@0
   324
		CFbsBitGc * gc = iScreen.GetBitGc();
sl@0
   325
		gc->SetPenColor(aColor);
sl@0
   326
		gc->SetPenStyle(CGraphicsContext::ESolidPen);
sl@0
   327
		gc->SetPenSize(TSize(2,2));
sl@0
   328
		gc->SetBrushColor(aFill);
sl@0
   329
		gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
sl@0
   330
		TRect smaller = *aRect;
sl@0
   331
		smaller.iBr.iX -= 1;
sl@0
   332
		smaller.iBr.iY -= 1;
sl@0
   333
		gc->DrawRect(smaller);
sl@0
   334
		iScreen.Update();
sl@0
   335
		}
sl@0
   336
	}
sl@0
   337
sl@0
   338
void CScreenRedraw::DebugRegion(TRgb aColor, TRgb aFill, const TRegion * aRegion)
sl@0
   339
	{
sl@0
   340
	if (aRegion)
sl@0
   341
		{
sl@0
   342
		CFbsBitGc * gc = iScreen.GetBitGc();
sl@0
   343
		gc->SetPenColor(aColor);
sl@0
   344
		gc->SetPenStyle(CGraphicsContext::ESolidPen);
sl@0
   345
		gc->SetPenSize(TSize(2,2));
sl@0
   346
		gc->SetBrushColor(aFill);
sl@0
   347
		gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
sl@0
   348
		for (const TRect *rect = aRegion->RectangleList(); rect - aRegion->RectangleList() < aRegion->Count(); ++rect)
sl@0
   349
			{
sl@0
   350
			TRect smaller = *rect;
sl@0
   351
			smaller.iBr.iX -= 1;
sl@0
   352
			smaller.iBr.iY -= 1;
sl@0
   353
			gc->DrawRect(smaller);
sl@0
   354
			}
sl@0
   355
		iScreen.Update();
sl@0
   356
		}
sl@0
   357
	}
sl@0
   358
#endif 
sl@0
   359
sl@0
   360
void CScreenRedraw::OnAnimation()
sl@0
   361
	{
sl@0
   362
	LOG_SCREEN_REDRAW_START
sl@0
   363
	ASSERT(!iAnimating);
sl@0
   364
	ASSERT(iScheduled);
sl@0
   365
	iAnimating = ETrue;
sl@0
   366
	iScheduled = EFalse;
sl@0
   367
	TBool futureAnimationRequired = EFalse;	
sl@0
   368
sl@0
   369
	CWsActiveScheduler::Static()->PrepareDraw();
sl@0
   370
	
sl@0
   371
	// Calculate any updates required by region changes:
sl@0
   372
	RegionUpdate();
sl@0
   373
sl@0
   374
	// Add the timed rectangles to the invalid region:
sl@0
   375
	iNow.UniversalTime();
sl@0
   376
	TInt count(iTimedDrawRect.Count());
sl@0
   377
	while (0 < count)
sl@0
   378
		{
sl@0
   379
		if(iTimedDrawRect[0].iTime.Int64() <= iNow.Int64())
sl@0
   380
			{
sl@0
   381
			iInvalid.AddRect(iTimedDrawRect[0].iRect);
sl@0
   382
			iTimedDrawRect.Remove(0);
sl@0
   383
			count--;
sl@0
   384
			}
sl@0
   385
		else
sl@0
   386
			{
sl@0
   387
			futureAnimationRequired = ETrue;	
sl@0
   388
			break;
sl@0
   389
			}
sl@0
   390
		}
sl@0
   391
sl@0
   392
	// Animating rectangles could cause iInvalid to overlap iTopLayer,
sl@0
   393
	// in which case iTopLayer won't work.
sl@0
   394
	iTopLayer.SubRegion(iInvalid);
sl@0
   395
	iTopLayer.Intersect(iScreen.RootWindow()->WindowArea());
sl@0
   396
	iTopLayer.Tidy();
sl@0
   397
	// Anything in the top layer is implcitly invalid
sl@0
   398
	iInvalid.Union(iTopLayer);
sl@0
   399
	iInvalid.Intersect(iScreen.RootWindow()->WindowArea());
sl@0
   400
	
sl@0
   401
/*
sl@0
   402
	if (const CDebugBar* dbg = iScreen.DebugBar())
sl@0
   403
		{
sl@0
   404
		iTopLayer.SubRect(dbg->Rect());
sl@0
   405
		iInvalid.SubRect(dbg->Rect());
sl@0
   406
		}
sl@0
   407
*/
sl@0
   408
sl@0
   409
	iInvalid.Tidy();
sl@0
   410
sl@0
   411
	if(iInvalid.CheckError()) //Will: agree with Andy, want bounding rects instead!
sl@0
   412
		{
sl@0
   413
		iTopLayer.Clear();
sl@0
   414
		iInvalid.Clear();
sl@0
   415
		iInvalid.Copy(iScreen.RootWindow()->WindowArea()); // assumed cannot fail, all regions can contain at least 1 rect..
sl@0
   416
		}
sl@0
   417
sl@0
   418
	iInvalid.SubRegion( iBannedRegion );	
sl@0
   419
	iInvalid.Tidy();
sl@0
   420
	iTopLayer.SubRegion( iBannedRegion );
sl@0
   421
	iTopLayer.Tidy();
sl@0
   422
	
sl@0
   423
	STACK_REGION invalidCopy;
sl@0
   424
	invalidCopy.Copy(iInvalid);
sl@0
   425
	TWalkWindowTreeScheduleRegions regionScheduler(&invalidCopy, iTopLayer);
sl@0
   426
	TWalkWindowTreeScheduleFallback fallbackScheduler(iScreen.FallbackMap());
sl@0
   427
	TWalkWindowTreeSchedule * scheduler = &regionScheduler;
sl@0
   428
	
sl@0
   429
	// At this point, if the DEBUG_REGION is being used:
sl@0
   430
	// Red represents invalid regions that need to be redrawn completely.
sl@0
   431
	// Yellow represents regions that only need the top window to be drawn.
sl@0
   432
	// Blue represents regions which are being animated server side.
sl@0
   433
	if (iRenderScheduled || !iInvalid.IsEmpty())
sl@0
   434
		{
sl@0
   435
		iRenderScheduled = EFalse;
sl@0
   436
		// invalidCopy.ForceError(); //### DEBUG
sl@0
   437
		
sl@0
   438
		iScreen.RootWindow()->WalkWindowTree(regionScheduler,EWalkChildren);
sl@0
   439
		if (!regionScheduler.ScheduledRegionsOk())
sl@0
   440
			{
sl@0
   441
			// our region calculations for what to draw failed at some point.
sl@0
   442
			// From this point on we MUST NOT rely on allocating memory
sl@0
   443
			// Andy - localRedrawRegion allocates
sl@0
   444
			// Andy - setPenSize allocates (even if you don't call it)
sl@0
   445
			// Andy - all draw commands add to a gdi dirty region (which allocates)
sl@0
   446
			// Andy - combining client clipping regions with window clipping regions allocates
sl@0
   447
			scheduler = &fallbackScheduler;
sl@0
   448
			iScreen.FallbackMap()->Prepare();
sl@0
   449
			iScreen.RootWindow()->WalkWindowTree(fallbackScheduler,EWalkChildren);
sl@0
   450
			}
sl@0
   451
		
sl@0
   452
		CWsActiveScheduler::Static()->StartDraw();
sl@0
   453
		CWsMemoryManager::Static()->EnableReserve();
sl@0
   454
sl@0
   455
		if (&fallbackScheduler == scheduler)
sl@0
   456
			iAnimationRegion = iScreen.FallbackMap()->Region();
sl@0
   457
		else
sl@0
   458
			iAnimationRegion = &iInvalid;
sl@0
   459
		
sl@0
   460
		// Redraw debug regions more brightly than before:
sl@0
   461
		DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x80),&iInvalid);
sl@0
   462
		DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x80),&iTopLayer);
sl@0
   463
		
sl@0
   464
		RWsRegion accumulatedDrawing;
sl@0
   465
		
sl@0
   466
		// Pipe the drawing into the first render stage:
sl@0
   467
		CFbsBitGc * stageGc = iRenderStages->Begin();
sl@0
   468
		
sl@0
   469
		for (CWsWindow * win = scheduler->HeadWindow(); win; win = win->NextScheduled())
sl@0
   470
			{
sl@0
   471
			const TRegion * targetRegion = scheduler->Region(win);
sl@0
   472
			const TRect * screenRect = 0;
sl@0
   473
			if ((&fallbackScheduler == scheduler) && !targetRegion->IsContainedBy(iScreen.RootWindow()->Abs()))
sl@0
   474
				{
sl@0
   475
				screenRect = &iScreen.RootWindow()->Abs();
sl@0
   476
				}
sl@0
   477
			if (!screenRect)
sl@0
   478
				{
sl@0
   479
				// Purple regions are about to be drawn
sl@0
   480
				DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80),targetRegion);
sl@0
   481
				// Do the drawing
sl@0
   482
				stageGc->Reset();
sl@0
   483
				win->Render(stageGc, *targetRegion);
sl@0
   484
				accumulatedDrawing.Union(*targetRegion);
sl@0
   485
				// Green regions have been drawn
sl@0
   486
				DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80),targetRegion);
sl@0
   487
				}
sl@0
   488
			else
sl@0
   489
				{
sl@0
   490
				// Our region overlaps the edges of the screen, and we have no memory
sl@0
   491
				// to create a clipped one, so we will use a single-rect region for each rect
sl@0
   492
				// and call Draw multiple times.
sl@0
   493
				TRegionFix<1> rectRegion;
sl@0
   494
				for (const TRect * rect = targetRegion->RectangleList() + targetRegion->Count() - 1; rect >= targetRegion->RectangleList(); --rect)
sl@0
   495
					{
sl@0
   496
					rectRegion.Clear();
sl@0
   497
					TRect combined(*screenRect);
sl@0
   498
					combined.Intersection(*rect);
sl@0
   499
					rectRegion.AddRect(combined);
sl@0
   500
					// Purple regions are about to be drawn
sl@0
   501
					DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80),&rectRegion);
sl@0
   502
					// Do the drawing
sl@0
   503
					stageGc->Reset();
sl@0
   504
					win->Render(stageGc, rectRegion);
sl@0
   505
					accumulatedDrawing.AddRect(combined);
sl@0
   506
					// Green regions have been drawn
sl@0
   507
					DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80),&rectRegion);
sl@0
   508
					}
sl@0
   509
				}
sl@0
   510
			DEBUGOSB
sl@0
   511
			}
sl@0
   512
sl@0
   513
		DEBUGOSB			
sl@0
   514
		iScreen.SpriteManager()->DrawFloatingSprites(stageGc,accumulatedDrawing); // we limit sprite drawing over only actually affected ares but necessary to all "planned" for redraw
sl@0
   515
		if (!accumulatedDrawing.CheckError() && iScreen.SpriteManager()->SpriteCount() == 0)
sl@0
   516
			{
sl@0
   517
			iAnimationRegion = &accumulatedDrawing;
sl@0
   518
			}
sl@0
   519
sl@0
   520
		// Tell the render stage we've finished:
sl@0
   521
		iRenderStages->End();
sl@0
   522
		
sl@0
   523
		// We nolonger need the regions
sl@0
   524
		for (CWsWindow * win = scheduler->HeadWindow(); win; win = win->NextScheduled())
sl@0
   525
			{
sl@0
   526
			win->ClearScheduledRegion();
sl@0
   527
			}
sl@0
   528
			
sl@0
   529
		CWsMemoryManager::Static()->DisableReserve();
sl@0
   530
sl@0
   531
		if (const CDebugBar* dbg = iScreen.DebugBar())
sl@0
   532
			{
sl@0
   533
			dbg->RedrawDebugBar();
sl@0
   534
			}
sl@0
   535
sl@0
   536
		// At this point, if the DEBUG_REGION is being used, there should usually be only green regions
sl@0
   537
		// displayed.  If we see red or yellow, then something didn't get redrawn that should have been.
sl@0
   538
		// If we see purple then a window has disobeyed the const setting on the region to draw.
sl@0
   539
		// Red or yellow is valid - a window can decide it isn't ready to draw yet - purple is very bad.
sl@0
   540
		iScreen.Update();
sl@0
   541
		// At this point, if the DEBUG_REGION is being used, there should be no regions visible
sl@0
   542
		// of any colour.  If we see green, then it means an area of the screen was drawn to which
sl@0
   543
		// wasn't invalid, or the screen update call failed.  The former is more likely.
sl@0
   544
		// If we still see red or yellow it is a region that is not yet ready to draw.
sl@0
   545
sl@0
   546
		const TRect* rect = iAnimationRegion->RectangleList();
sl@0
   547
		TInt pixels = 0;
sl@0
   548
		for(TInt r = iAnimationRegion->Count(); r>0; r--)
sl@0
   549
			{
sl@0
   550
			pixels += (rect->Width()*rect->Height());
sl@0
   551
			rect++;
sl@0
   552
			}
sl@0
   553
		CWsActiveScheduler::Static()->StopDraw(pixels);
sl@0
   554
		
sl@0
   555
		TWindowServerEvent::NotifyScreenDrawingEvent(iAnimationRegion);
sl@0
   556
sl@0
   557
		iAnimationRegion = 0;
sl@0
   558
		accumulatedDrawing.Close();
sl@0
   559
		iInvalid.Clear();
sl@0
   560
		}
sl@0
   561
	else
sl@0
   562
		{
sl@0
   563
		CWsActiveScheduler::Static()->CancelPrepare();
sl@0
   564
		}
sl@0
   565
sl@0
   566
	iInvalid.Clear();
sl@0
   567
	iTopLayer.Clear();
sl@0
   568
	invalidCopy.Close();
sl@0
   569
	iAnimating = EFalse;
sl@0
   570
	
sl@0
   571
	if (futureAnimationRequired && iTimedDrawRect.Count() > 0 && !iScheduled)
sl@0
   572
		{
sl@0
   573
		// If this flag is set then it means there were already some animations scheduled when we ran,
sl@0
   574
		// but they themselves didn't run.  We need to make sure we have _something_ scheduled.
sl@0
   575
		CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen, iTimedDrawRect[0].iTime);
sl@0
   576
		iScheduled = ETrue;
sl@0
   577
		}
sl@0
   578
	
sl@0
   579
	if(iObserver)
sl@0
   580
		{
sl@0
   581
		iObserver->ScreenUpdated(iScreen.ScreenNumber());
sl@0
   582
		iObserver=NULL; //once signalled we are never going to call it again
sl@0
   583
		}
sl@0
   584
	LOG_SCREEN_REDRAW_END
sl@0
   585
	}
sl@0
   586
sl@0
   587
//
sl@0
   588
void CScreenRedraw::DiscardAllSchedules()
sl@0
   589
	{
sl@0
   590
	ASSERT(!iAnimating);
sl@0
   591
	
sl@0
   592
	iTimedDrawRect.Reset();
sl@0
   593
	iInvalid.Clear();
sl@0
   594
	iInvalid.Tidy();
sl@0
   595
	}
sl@0
   596
sl@0
   597
/**
sl@0
   598
Indicates that a window has moved or changed ordinal position so that the visible regions
sl@0
   599
of all windows needs to be recalculated
sl@0
   600
*/
sl@0
   601
void CScreenRedraw::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
sl@0
   602
	{
sl@0
   603
	iRegionUpdateScheduled = ETrue;
sl@0
   604
	ScheduleRedraw();
sl@0
   605
	if(aDefinitelyDirty)
sl@0
   606
		{
sl@0
   607
		iInvalid.Union(*aDefinitelyDirty);
sl@0
   608
		// Cyan regions for invalidations caused by this:
sl@0
   609
		DEBUG_REGION(TRgb(0x00, 0xFF, 0xFF),TRgb(0x00, 0xFF, 0xFF, 0x20),aDefinitelyDirty);
sl@0
   610
		}
sl@0
   611
	}
sl@0
   612
	
sl@0
   613
/**
sl@0
   614
Recalculates visible regions and schedules redraws or queues redraw events in response to
sl@0
   615
any changes
sl@0
   616
*/
sl@0
   617
void CScreenRedraw::RegionUpdate()
sl@0
   618
	{
sl@0
   619
	if (iRegionUpdateScheduled)
sl@0
   620
		{
sl@0
   621
		iRegionUpdateScheduled = EFalse;
sl@0
   622
		
sl@0
   623
		TWalkWindowTreeUpdateRegions updater(iScreen);
sl@0
   624
		updater.Walk();
sl@0
   625
sl@0
   626
		WsPointer::ReLogCurrentWindow();
sl@0
   627
		CWsTop::TriggerRedraws(iScreen.RootWindow());
sl@0
   628
		}
sl@0
   629
	}
sl@0
   630
sl@0
   631
void CScreenRedraw::SetObserver(MWsScreenRedrawObserver* aObserver)
sl@0
   632
	{
sl@0
   633
	iObserver = aObserver;
sl@0
   634
	}
sl@0
   635
sl@0
   636
TBool CScreenRedraw::IsUpdatePending()
sl@0
   637
	{
sl@0
   638
	if(iScheduled || iAnimating)
sl@0
   639
		return ETrue;
sl@0
   640
	else 
sl@0
   641
		return EFalse;
sl@0
   642
	}
sl@0
   643
sl@0
   644
/**
sl@0
   645
 Overidding MWsObjectProvider
sl@0
   646
*/
sl@0
   647
TAny* CScreenRedraw::ResolveObjectInterface(TUint aTypeId)
sl@0
   648
	{
sl@0
   649
	TAny* interface = NULL;
sl@0
   650
sl@0
   651
	switch (aTypeId)
sl@0
   652
		{
sl@0
   653
		case KWsScreenRedraw:
sl@0
   654
			interface = static_cast<MWsScreenRedraw*>(this);
sl@0
   655
			break;
sl@0
   656
		}
sl@0
   657
sl@0
   658
	if (!interface)
sl@0
   659
		{
sl@0
   660
		interface = iRenderStages->ResolveObjectInterface(aTypeId);
sl@0
   661
		}
sl@0
   662
	
sl@0
   663
	return interface;
sl@0
   664
	}
sl@0
   665
sl@0
   666
const TRegion * CScreenRedraw::AnimationRegion() const
sl@0
   667
	{
sl@0
   668
	if (iAnimating)
sl@0
   669
		return iAnimationRegion;
sl@0
   670
	else
sl@0
   671
		return 0;
sl@0
   672
	}
sl@0
   673
sl@0
   674
void CScreenRedraw::UpdateDevice()
sl@0
   675
	{
sl@0
   676
	iScreen.Update();
sl@0
   677
	}
sl@0
   678
sl@0
   679
void CScreenRedraw::BanThisRegionUpdate( TRegion& aForbiddenRegion )
sl@0
   680
	{
sl@0
   681
	iBannedRegion.Union( aForbiddenRegion );
sl@0
   682
	iBannedRegion.Tidy();
sl@0
   683
	}
sl@0
   684
sl@0
   685
void CScreenRedraw::LiftRegionUpdateBan( TRegion& aFormerForbiddenRegion  )
sl@0
   686
	{
sl@0
   687
	iBannedRegion.SubRegion( aFormerForbiddenRegion );
sl@0
   688
	iBannedRegion.Tidy();
sl@0
   689
	}
sl@0
   690
sl@0
   691
void CScreenRedraw::AcceptFadeRequest( CWsWindow* aWin, TBool aFadeOn, TBool aFadeBehind, TBool aIncludeChildren )
sl@0
   692
	{
sl@0
   693
	if ( aFadeOn )
sl@0
   694
		{
sl@0
   695
		TWalkWindowTreeScheduleFadeNoRedraw walkerFadeNoRedraw;
sl@0
   696
		if ( aFadeBehind )
sl@0
   697
			{
sl@0
   698
			aWin->WalkWindowTree( walkerFadeNoRedraw, EWalkBehind );				
sl@0
   699
			}
sl@0
   700
		else
sl@0
   701
			{
sl@0
   702
			if(aWin->WinType() != EWinTypeGroup)
sl@0
   703
				{
sl@0
   704
				ScheduleRegionUpdate( aWin->VisibleRegionIfValid() );
sl@0
   705
				}
sl@0
   706
			
sl@0
   707
			if ( aIncludeChildren )
sl@0
   708
				{
sl@0
   709
				aWin->WalkWindowTree( walkerFadeNoRedraw, EWalkChildren );
sl@0
   710
				}				
sl@0
   711
			}
sl@0
   712
		}
sl@0
   713
	else
sl@0
   714
		{	// fade off, just initiate redraw
sl@0
   715
		TWalkWindowTreeScheduleRedraws walkerRedraw( TWalkWindowTreeScheduleRedraws::ERedrawFilterOmitDSA ) ;
sl@0
   716
		if ( aFadeBehind )
sl@0
   717
			{
sl@0
   718
			aWin->WalkWindowTree( walkerRedraw, EWalkBehind );								
sl@0
   719
			}
sl@0
   720
		else
sl@0
   721
			{ // fade this win not behind
sl@0
   722
			if ( !aWin->IsDSAHost() )
sl@0
   723
				{
sl@0
   724
				AddRedrawRegion( aWin->VisibleRegion() );
sl@0
   725
				}
sl@0
   726
			if ( aIncludeChildren )
sl@0
   727
				{
sl@0
   728
				aWin->WalkWindowTree( walkerRedraw, EWalkChildren );
sl@0
   729
				}				
sl@0
   730
			}			
sl@0
   731
		}		
sl@0
   732
	ScheduleRegionUpdate(NULL);
sl@0
   733
	}