os/graphics/windowing/windowserver/nga/SERVER/openwfc/ScreenRedraw.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) 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
//
sl@0
    15
sl@0
    16
#include "ScreenRedraw.h"
sl@0
    17
sl@0
    18
#include <hal.h>
sl@0
    19
sl@0
    20
#include "debugbar.h"
sl@0
    21
#include "inifile.h"
sl@0
    22
#include "screen.h"
sl@0
    23
#include "pointer.h"
sl@0
    24
#include "rootwin.h"
sl@0
    25
#include "walkwindowtree.h"
sl@0
    26
#include "wstop.h"
sl@0
    27
#include "WsMemMgr.h"
sl@0
    28
#include "renderstagemanager.h"
sl@0
    29
#include "graphics/WsRenderStageFactory.h"
sl@0
    30
#include "graphics/WsRenderStage.h"
sl@0
    31
#include "graphics/wsgraphicscontext.h"
sl@0
    32
#include "EVENT.H"
sl@0
    33
sl@0
    34
#ifdef USE_DEBUG_FRAME_CAPTURE
sl@0
    35
#include <graphics/wsscreendevice.h>
sl@0
    36
#include "../debuglog/debuglog.h"
sl@0
    37
#endif
sl@0
    38
sl@0
    39
GLREF_D CDebugLogBase *wsDebugLog;
sl@0
    40
sl@0
    41
#ifdef USE_DEBUG_REGIONS
sl@0
    42
#	define DEBUG_REGION(col,fill,reg) DebugRegion(col,fill,reg)
sl@0
    43
#	define DEBUG_RECT(col,fill,rect) DebugRect(col,fill,rect)
sl@0
    44
#else
sl@0
    45
#	define DEBUG_REGION(col,fill,reg)
sl@0
    46
#	define DEBUG_RECT(col,fill,rect)
sl@0
    47
#endif
sl@0
    48
sl@0
    49
#ifdef _DEBUG
sl@0
    50
# define LOG_SCREEN_REDRAW_START {if (wsDebugLog) {_LIT(KLogScreenRedrawStart, ">> CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawStart);}}
sl@0
    51
# define LOG_SCREEN_REDRAW_END {if (wsDebugLog) {_LIT(KLogScreenRedrawEnd, "<< CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawEnd);}}
sl@0
    52
#else
sl@0
    53
# define LOG_SCREEN_REDRAW_START
sl@0
    54
# define LOG_SCREEN_REDRAW_END
sl@0
    55
#endif
sl@0
    56
sl@0
    57
sl@0
    58
#ifdef USE_DEBUG_FRAME_CAPTURE
sl@0
    59
#	ifdef __WINS__
sl@0
    60
	_LIT(KDefaultFrameCaptureLocation, "c:\\");
sl@0
    61
#	else
sl@0
    62
	_LIT(KDefaultFrameCaptureLocation, "e:\\");
sl@0
    63
#	endif
sl@0
    64
#endif //USE_DEBUG_FRAME_CAPTURE
sl@0
    65
sl@0
    66
CScreenRedraw::TTimedRect::TTimedRect(TAnimType aType, const TRect& aRect, const TTime& aTime, CWsWindow* aWindow)
sl@0
    67
	: iType(aType), iRect(aRect), iTime(aTime), iWindow(aWindow)
sl@0
    68
	{
sl@0
    69
	}
sl@0
    70
sl@0
    71
TInt CScreenRedraw::TTimedRect::Compare(const TTimedRect& aOne,const TTimedRect& aOther)
sl@0
    72
	{
sl@0
    73
	if(aOne.iTime < aOther.iTime)
sl@0
    74
		return -1;
sl@0
    75
	else if(aOne.iTime > aOther.iTime)
sl@0
    76
		return 1;
sl@0
    77
	else
sl@0
    78
		return 0;
sl@0
    79
	}
sl@0
    80
sl@0
    81
CScreenRedraw * CScreenRedraw::NewL(CScreen& aScreen)
sl@0
    82
	{
sl@0
    83
	CScreenRedraw * self = new (ELeave) CScreenRedraw(aScreen);
sl@0
    84
	CleanupStack::PushL(self);
sl@0
    85
	self->ConstructL();
sl@0
    86
	CleanupStack::Pop(self);
sl@0
    87
	return self;
sl@0
    88
	}
sl@0
    89
	
sl@0
    90
CScreenRedraw::CScreenRedraw(CScreen& aScreen): iScreen(aScreen)
sl@0
    91
	{
sl@0
    92
	}
sl@0
    93
sl@0
    94
CScreenRedraw::~CScreenRedraw()
sl@0
    95
	{
sl@0
    96
	CRenderStageManager::Release(iRenderStages);
sl@0
    97
	iRenderStages = NULL;
sl@0
    98
	iTimedDrawRect.Close();
sl@0
    99
	iInvalid.Reset();
sl@0
   100
	iTopElement.Reset();
sl@0
   101
	iQuickFadeList.Reset();
sl@0
   102
#ifdef USE_DEBUG_FRAME_CAPTURE
sl@0
   103
	delete iDebugBitmap;
sl@0
   104
	delete iDebugBitmapDevice;
sl@0
   105
	delete iDebugBitmapGc;
sl@0
   106
#endif //USE_DEBUG_FRAME_CAPTURE
sl@0
   107
	}
sl@0
   108
sl@0
   109
void CScreenRedraw::ConstructL()
sl@0
   110
	{
sl@0
   111
	
sl@0
   112
	LEAVE_LOG_INSTALL_C;
sl@0
   113
	
sl@0
   114
	iRenderStages = CRenderStageManager::ConnectL(
sl@0
   115
						iScreen.ScreenNumber(),
sl@0
   116
						static_cast<MWsScreen*>(&iScreen),
sl@0
   117
						this
sl@0
   118
						);
sl@0
   119
	
sl@0
   120
	LEAVE_LOG_UNINSTALL_C;
sl@0
   121
	
sl@0
   122
	WS_ASSERT_ALWAYS(iRenderStages, EWsPanicNoRenderStagePipeline);
sl@0
   123
	iRenderStageTextCursor = static_cast<MWsTextCursor*>(iRenderStages->ResolveObjectInterface(KMWsTextCursor));
sl@0
   124
	WS_ASSERT_ALWAYS(iRenderStageTextCursor, EWsPanicTextCursorInterfaceMissing);
sl@0
   125
sl@0
   126
#ifdef USE_DEBUG_FRAME_CAPTURE
sl@0
   127
	_LIT(KWSERVIniFileVarFrameCapture,"FRAMECAPTURE");
sl@0
   128
	iFrameCapture = WsIniFile->FindVar(KWSERVIniFileVarFrameCapture);
sl@0
   129
sl@0
   130
	// Location to save captured images
sl@0
   131
	if (!WsIniFile->FindVar(KWSERVIniFileVarFrameCapture, iFrameCaptureLocation) || iFrameCaptureLocation.Length() == 0)
sl@0
   132
		{
sl@0
   133
		iFrameCaptureLocation.Set(KDefaultFrameCaptureLocation);
sl@0
   134
		}
sl@0
   135
#endif //USE_DEBUG_FRAME_CAPTURE
sl@0
   136
	}
sl@0
   137
sl@0
   138
MWsTextCursor* CScreenRedraw::RenderStageTextCursor() const
sl@0
   139
	{
sl@0
   140
	return iRenderStageTextCursor;
sl@0
   141
	}
sl@0
   142
sl@0
   143
const TTime& CScreenRedraw::Now() const
sl@0
   144
	{
sl@0
   145
	if(!iAnimating)
sl@0
   146
		{
sl@0
   147
		iNow.UniversalTime();
sl@0
   148
		}
sl@0
   149
	return iNow;
sl@0
   150
	}
sl@0
   151
sl@0
   152
void CScreenRedraw::ScheduleRender(const TTimeIntervalMicroSeconds& aFromNow)
sl@0
   153
	{
sl@0
   154
	iRenderScheduled = ETrue;
sl@0
   155
	TTime then(Now() + aFromNow);
sl@0
   156
	if ((!iScheduled) || then < iNext)
sl@0
   157
		iNext = then;
sl@0
   158
	iScheduled = ETrue;
sl@0
   159
	CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
sl@0
   160
	}
sl@0
   161
sl@0
   162
void CScreenRedraw::ScheduleRedraw()
sl@0
   163
	{
sl@0
   164
	iNext = Now();
sl@0
   165
	iScheduled = ETrue;
sl@0
   166
		
sl@0
   167
	// The other scheduler also removes future animations which this one encompasses.
sl@0
   168
	// We choose not to do the region calculations needed to achieve that here.
sl@0
   169
	CWindowServer* server = CWsTop::WindowServer();
sl@0
   170
	ASSERT(server);
sl@0
   171
	MWsAnimationScheduler* sched = server->AnimationScheduler();
sl@0
   172
	if(sched)
sl@0
   173
		sched->ScheduleRedraw(iScreen,iNext);
sl@0
   174
	else
sl@0
   175
		RDebug::Printf("CWsTop::WindowServer()->RedrawScheduler() is NULL!!!");			
sl@0
   176
	}
sl@0
   177
sl@0
   178
/**
sl@0
   179
@param aRect in screen coordinates 
sl@0
   180
*/
sl@0
   181
void CScreenRedraw::ScheduleAnimation(TAnimType aType, const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow,const TTimeIntervalMicroSeconds& /*aFreq*/,const TTimeIntervalMicroSeconds& /*aStop*/, CWsWindow* aWindow)
sl@0
   182
	{
sl@0
   183
	TRect scheduledRect(aRect);
sl@0
   184
	TRect test(aRect);
sl@0
   185
	// In changetracking mode, animation is either scheduled via a window dirty region (aWindow non-NULL)
sl@0
   186
	// or via a sprite manager dirty region (aWindow NULL) if scheduling a floating sprite
sl@0
   187
	if (iScreen.ChangeTracking())
sl@0
   188
		{
sl@0
   189
		if (aWindow)
sl@0
   190
			{
sl@0
   191
			// scheduling a window dirty rect for a window/anim/sprite
sl@0
   192
			test.Intersection(aWindow->Abs());
sl@0
   193
			scheduledRect.Move(-aWindow->Origin()); // convert to window coordinates
sl@0
   194
			}
sl@0
   195
		// else, // scheduling a sprite manager dirty rect for a floating sprite
sl@0
   196
		}
sl@0
   197
	else
sl@0
   198
		{
sl@0
   199
		// scheduling a screen dirty rect
sl@0
   200
		test.Intersection(iScreen.DrawableArea());
sl@0
   201
		aWindow = NULL; // ensure all future timed draw screen rects are checked below
sl@0
   202
		}
sl@0
   203
	if(!test.IsEmpty())
sl@0
   204
		{
sl@0
   205
		const TTime then(Now() + aFromNow);
sl@0
   206
		TTimedRect tRect(aType, scheduledRect, then, aWindow);
sl@0
   207
sl@0
   208
		const TInt error = iTimedDrawRect.InsertInOrderAllowRepeats(tRect,TTimedRect::Compare);
sl@0
   209
		if (KErrNone == error)
sl@0
   210
	   		{
sl@0
   211
			if (iScheduled)
sl@0
   212
				{
sl@0
   213
				if (then < iNext)
sl@0
   214
					{
sl@0
   215
					iNext = then;
sl@0
   216
					}
sl@0
   217
				}
sl@0
   218
			else
sl@0
   219
				{
sl@0
   220
				iNext = then;
sl@0
   221
				iScheduled = ETrue;
sl@0
   222
				}
sl@0
   223
			// remove further futures that are completely contained
sl@0
   224
			TInt count = iTimedDrawRect.Count();
sl@0
   225
			for(TInt i=0; i<count; i++)
sl@0
   226
				{
sl@0
   227
				const TTimedRect& future = iTimedDrawRect[i];
sl@0
   228
				if (future.iWindow != aWindow) // only check futures for the window/floating sprite/screen we're scheduling
sl@0
   229
					continue;
sl@0
   230
				if(future.iTime.Int64() > then.Int64())
sl@0
   231
					{
sl@0
   232
					TRect rect(scheduledRect);
sl@0
   233
					rect.BoundingRect(future.iRect);
sl@0
   234
					if(rect == scheduledRect) // future is completely contained within scheduledRect
sl@0
   235
						{
sl@0
   236
						iTimedDrawRect.Remove(i);
sl@0
   237
						count--;
sl@0
   238
						i--;
sl@0
   239
						}
sl@0
   240
					}
sl@0
   241
				}
sl@0
   242
			CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
sl@0
   243
sl@0
   244
			// Blue rectangles for scheduled animations
sl@0
   245
			DEBUG_RECT(TRgb(0x00, 0x00, 0xFF),TRgb(0x00, 0x00, 0xFF, 0x20),&aRect);
sl@0
   246
			}
sl@0
   247
		}
sl@0
   248
	}
sl@0
   249
sl@0
   250
TBool CScreenRedraw::IsScheduled(TAnimType aType, const TRect& aRect, CWsWindow* aWindow) const
sl@0
   251
	{
sl@0
   252
	TRect rect(aRect);
sl@0
   253
	if(aWindow)
sl@0
   254
		{
sl@0
   255
		rect.Move(-aWindow->Origin()); //convert to window coordinates
sl@0
   256
		}
sl@0
   257
	
sl@0
   258
	const TInt count(iTimedDrawRect.Count());
sl@0
   259
	for(TInt i=0; i<count; i++)
sl@0
   260
		{
sl@0
   261
		if ((iTimedDrawRect[i].iType == aType) && 
sl@0
   262
				(iTimedDrawRect[i].iRect == rect) && 
sl@0
   263
				(iTimedDrawRect[i].iWindow == aWindow))
sl@0
   264
			{
sl@0
   265
			return ETrue;
sl@0
   266
			}
sl@0
   267
		}
sl@0
   268
	return EFalse;
sl@0
   269
	}
sl@0
   270
sl@0
   271
// This adds a region to the stored invalid region.
sl@0
   272
// The invalid region is the area of the screen that needs to be redrawn in addition to any animations.
sl@0
   273
// The draw region is the area of the screen on which only the top window needs to be redrawn.
sl@0
   274
// If the top window has transparency, this can only be true when it has been made newly visible.
sl@0
   275
// The value of aSchedule could be determined automatically from iAnimating, but passing it this way
sl@0
   276
// allows us to have the assert, which is a very valuable assert.
sl@0
   277
void CScreenRedraw::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
sl@0
   278
	{
sl@0
   279
	WS_ASSERT_DEBUG(!iAnimating || !aSchedule, EWsPanicScheduledRedraw);
sl@0
   280
sl@0
   281
	if(aRegion.CheckError())
sl@0
   282
		{
sl@0
   283
		iInvalid.ForceError();
sl@0
   284
sl@0
   285
		if (aSchedule)
sl@0
   286
			ScheduleRedraw();
sl@0
   287
		}
sl@0
   288
	else if(aRegion.Count()) // often called despite window not being visible
sl@0
   289
		{
sl@0
   290
		if (aDepth == ERedrawAll)
sl@0
   291
			{
sl@0
   292
			// red lines for an invalid region which is ready to be drawn
sl@0
   293
			DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x20),&aRegion);
sl@0
   294
			iInvalid.Union(aRegion);
sl@0
   295
			
sl@0
   296
			if (aSchedule)
sl@0
   297
				ScheduleRedraw();
sl@0
   298
			}
sl@0
   299
		else
sl@0
   300
			{
sl@0
   301
			// yellow lines for a valid region which we will draw on top of
sl@0
   302
			DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x20),&aRegion);
sl@0
   303
				
sl@0
   304
			iTopElement.Union(aRegion);
sl@0
   305
			
sl@0
   306
			if (aSchedule)
sl@0
   307
				ScheduleRedraw();			
sl@0
   308
			}
sl@0
   309
		}
sl@0
   310
	}
sl@0
   311
sl@0
   312
// This causes any asynchronously scheduled redraw to happen immediately
sl@0
   313
// It should be avoided where possible for performance reasons, but is 
sl@0
   314
// needed whenever the redraw store is discarded for a window which still
sl@0
   315
// has a redraw region pending.
sl@0
   316
// @note This method always attempts to DoRedrawNow(), even if currently
sl@0
   317
//		 animating. In this context, animation is a frame update. With NGA
sl@0
   318
// 		 that update completes at some point in the future. If you get a request
sl@0
   319
//		 to DoRedrawNow() the expectation is that all updates scheduled on or 
sl@0
   320
//		 before that point in time will have completed by the time the call 
sl@0
   321
//		 returns. An update may have been scheduled during a current (asynchronous)
sl@0
   322
//		 Animate call...the animation scheduler has this knowledge, so let it
sl@0
   323
//		 decide what to do.
sl@0
   324
sl@0
   325
void CScreenRedraw::DoRedrawNow()
sl@0
   326
	{
sl@0
   327
	CWsTop::WindowServer()->AnimationScheduler()->DoRedrawNow(iScreen);
sl@0
   328
	}
sl@0
   329
	
sl@0
   330
#ifdef USE_DEBUG_REGIONS
sl@0
   331
void CScreenRedraw::DebugRect(TRgb aColor, TRgb aFill, const TRect* aRect)
sl@0
   332
	{
sl@0
   333
	if (aRect)
sl@0
   334
		{
sl@0
   335
		CFbsBitGc * gc = iScreen.GetBitGc();
sl@0
   336
		gc->SetPenColor(aColor);
sl@0
   337
		gc->SetPenStyle(CGraphicsContext::ESolidPen);
sl@0
   338
		gc->SetPenSize(TSize(2,2));
sl@0
   339
		gc->SetBrushColor(aFill);
sl@0
   340
		gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
sl@0
   341
		TRect smaller = *aRect;
sl@0
   342
		smaller.iBr.iX -= 1;
sl@0
   343
		smaller.iBr.iY -= 1;
sl@0
   344
		gc->DrawRect(smaller);
sl@0
   345
		iScreen.Update();
sl@0
   346
		}
sl@0
   347
	}
sl@0
   348
sl@0
   349
void CScreenRedraw::DebugRegion(TRgb aColor, TRgb aFill, const TRegion * aRegion)
sl@0
   350
	{
sl@0
   351
	if (aRegion)
sl@0
   352
		{
sl@0
   353
		CFbsBitGc * gc = iScreen.GetBitGc();
sl@0
   354
		gc->SetPenColor(aColor);
sl@0
   355
		gc->SetPenStyle(CGraphicsContext::ESolidPen);
sl@0
   356
		gc->SetPenSize(TSize(2,2));
sl@0
   357
		gc->SetBrushColor(aFill);
sl@0
   358
		gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
sl@0
   359
		for (const TRect *rect = aRegion->RectangleList(); rect - aRegion->RectangleList() < aRegion->Count(); ++rect)
sl@0
   360
			{
sl@0
   361
			TRect smaller = *rect;
sl@0
   362
			smaller.iBr.iX -= 1;
sl@0
   363
			smaller.iBr.iY -= 1;
sl@0
   364
			gc->DrawRect(smaller);
sl@0
   365
			}
sl@0
   366
		iScreen.Update();
sl@0
   367
		}
sl@0
   368
	}
sl@0
   369
#endif 
sl@0
   370
sl@0
   371
void CScreenRedraw::OnAnimation(TRequestStatus* aFinished)
sl@0
   372
	{
sl@0
   373
	LOG_SCREEN_REDRAW_START
sl@0
   374
	WS_ASSERT_ALWAYS(!iAnimating,EWsPanicAnimationAlreadyAnimating);
sl@0
   375
	WS_ASSERT_ALWAYS(iScheduled,EWsPanicAnimationNotScheduled);
sl@0
   376
	iAnimating = ETrue;
sl@0
   377
	iScheduled = EFalse;
sl@0
   378
sl@0
   379
	const TBool changeTracking = iScreen.ChangeTracking();
sl@0
   380
	
sl@0
   381
 	const TRegionFix<1> fallbackRegion(iScreen.RootWindow()->Abs());
sl@0
   382
sl@0
   383
	CWsActiveScheduler::Static()->PrepareDraw();
sl@0
   384
	
sl@0
   385
	// Calculate any updates required by region changes:
sl@0
   386
	RegionUpdate();
sl@0
   387
sl@0
   388
	// Use the timed rectangles to mark screen, window or floating sprite as dirty
sl@0
   389
	RWsRegionBuf<10> floatingSpriteDirtyRegion;
sl@0
   390
 	const TBool futureAnimationRequired = ScheduleTimedRects(floatingSpriteDirtyRegion);
sl@0
   391
 	if (floatingSpriteDirtyRegion.CheckError())
sl@0
   392
 		{
sl@0
   393
 		floatingSpriteDirtyRegion.Reset();
sl@0
   394
 		floatingSpriteDirtyRegion.Copy(fallbackRegion);
sl@0
   395
 		}
sl@0
   396
 	
sl@0
   397
 	TWalkWindowTreeSchedule* scheduler = NULL;
sl@0
   398
 	TWalkWindowListSchedule windowScheduler(iScheduledWindowList, iInvalid); //ChangeTracking
sl@0
   399
	if (!changeTracking)
sl@0
   400
		{
sl@0
   401
		// Animating rectangles could cause iInvalid to overlap iTopElement, in which case iTopElement won't work.
sl@0
   402
		iTopElement.SubRegion(iInvalid);
sl@0
   403
		iTopElement.Intersect(iScreen.RootWindow()->WindowArea());
sl@0
   404
		iTopElement.Tidy();
sl@0
   405
		// Anything in the top element is implicitly invalid
sl@0
   406
		iInvalid.Union(iTopElement);
sl@0
   407
		iInvalid.Intersect(iScreen.RootWindow()->WindowArea());
sl@0
   408
		iInvalid.Tidy();
sl@0
   409
		
sl@0
   410
		if(iInvalid.CheckError())
sl@0
   411
			{
sl@0
   412
			iTopElement.Reset();
sl@0
   413
			iInvalid.Reset();
sl@0
   414
			iInvalid.Copy(fallbackRegion);
sl@0
   415
			}
sl@0
   416
		}
sl@0
   417
	else 
sl@0
   418
		{
sl@0
   419
		// In ChangeTracking mode, iInvalid is only used to tell the render stage
sl@0
   420
		// which part of the screen needs updating.
sl@0
   421
		iInvalid.Reset();
sl@0
   422
		iInvalid.Copy(floatingSpriteDirtyRegion);
sl@0
   423
		scheduler = &windowScheduler;
sl@0
   424
		windowScheduler.WalkWindowList();
sl@0
   425
		CWsTop::TriggerRedraws(iScreen.RootWindow()); //In case WalkWindowList did queue a request for the client to provide draw commands
sl@0
   426
		iInvalid.Tidy();
sl@0
   427
		if(iInvalid.CheckError())
sl@0
   428
			{
sl@0
   429
			iInvalid.Reset();
sl@0
   430
			iInvalid.Copy(fallbackRegion);
sl@0
   431
			}
sl@0
   432
		}
sl@0
   433
		
sl@0
   434
	// At this point, if the DEBUG_REGION is being used:
sl@0
   435
	// Red represents invalid regions that need to be redrawn completely.
sl@0
   436
	// Yellow represents regions that only need the top window to be drawn.
sl@0
   437
	// Blue represents regions which are being animated server side.
sl@0
   438
	if (iRenderScheduled || !iInvalid.IsEmpty() || iQuickFadeList.Count() > 0)  
sl@0
   439
		{
sl@0
   440
		iRenderScheduled = EFalse;
sl@0
   441
		
sl@0
   442
		TWalkWindowTreeScheduleRegions regionScheduler(iInvalid, iTopElement); //!ChangeTeacking
sl@0
   443
		TWalkWindowTreeScheduleFallback fallbackScheduler(iScreen.FallbackMap());//!ChangeTeacking
sl@0
   444
sl@0
   445
		RWsRegionBuf<20> animationRegion;
sl@0
   446
		
sl@0
   447
		if (!changeTracking)
sl@0
   448
			{	
sl@0
   449
			animationRegion.Copy(iInvalid);
sl@0
   450
			AddQuickFadeableRegions(animationRegion);
sl@0
   451
			if (animationRegion.CheckError())
sl@0
   452
				{
sl@0
   453
				animationRegion.Reset();
sl@0
   454
				animationRegion.Copy(fallbackRegion);
sl@0
   455
				}
sl@0
   456
			iAnimationRegion = &animationRegion; //iAnimationRegion must be reset to NULL before the call stack unwinds.
sl@0
   457
			
sl@0
   458
			scheduler = &regionScheduler;
sl@0
   459
			iScreen.RootWindow()->WalkWindowTree(regionScheduler,EWalkChildren);
sl@0
   460
			if (!regionScheduler.ScheduledRegionsOk())
sl@0
   461
				{
sl@0
   462
				// our region calculations for what to draw failed at some point.
sl@0
   463
				// From this point on we MUST NOT rely on allocating memory
sl@0
   464
				scheduler = &fallbackScheduler;
sl@0
   465
				iScreen.FallbackMap()->Prepare();
sl@0
   466
				iScreen.RootWindow()->WalkWindowTree(fallbackScheduler,EWalkChildren);
sl@0
   467
				iAnimationRegion = iScreen.FallbackMap()->Region();
sl@0
   468
				}
sl@0
   469
			}
sl@0
   470
		else
sl@0
   471
			{
sl@0
   472
			iAnimationRegion = &iInvalid;
sl@0
   473
			}
sl@0
   474
		
sl@0
   475
		CWsActiveScheduler::Static()->StartDraw();
sl@0
   476
		CWsMemoryManager::Static()->EnableReserve();
sl@0
   477
		
sl@0
   478
		// Redraw debug regions more brightly than before:
sl@0
   479
		DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x80),iAnimationRegion);
sl@0
   480
		DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x80),&iTopElement);
sl@0
   481
		
sl@0
   482
		// Pipe the drawing into the first render stage:
sl@0
   483
		iRenderStages->Begin(iAnimationRegion);
sl@0
   484
		
sl@0
   485
		RWsRegionBuf<10> accumulatedDrawing;
sl@0
   486
		
sl@0
   487
		MWsGraphicsContext * stageGc = static_cast<MWsGraphicsContext*>(iRenderStages->ResolveObjectInterface(KMWsGraphicsContext));
sl@0
   488
		for (CWsWindow * win = scheduler->HeadWindow(); (win!=NULL); win = win->NextScheduled())
sl@0
   489
			{
sl@0
   490
			if(!changeTracking)
sl@0
   491
					accumulatedDrawing.Union(scheduler->WindowRegion(*win));
sl@0
   492
			
sl@0
   493
			Render(*win, *stageGc, *scheduler);
sl@0
   494
			}
sl@0
   495
sl@0
   496
		if(!changeTracking && !accumulatedDrawing.CheckError())
sl@0
   497
			{
sl@0
   498
			//Fade any region of the screen scheduled for fading which has not been redrawn,
sl@0
   499
			DoQuickFade(stageGc, accumulatedDrawing);
sl@0
   500
			}
sl@0
   501
		accumulatedDrawing.Reset();
sl@0
   502
		
sl@0
   503
		//We limit floating sprite drawing to regions already touched
sl@0
   504
		iScreen.SpriteManager()->DrawFloatingSprites(stageGc, iScreen.ChangeTracking() ? floatingSpriteDirtyRegion : *iAnimationRegion);
sl@0
   505
		
sl@0
   506
		iRenderStages->End(aFinished);
sl@0
   507
		stageGc = NULL; //we're not allowed to draw outside Begin()/End()
sl@0
   508
sl@0
   509
#if defined(__WINS__) && defined(_DEBUG)
sl@0
   510
		MWsDebugBar * debugBar = static_cast<MWsDebugBar*>(iRenderStages->ResolveObjectInterface(KMWsDebugBar));
sl@0
   511
		if (debugBar) //optional for the licensees
sl@0
   512
			{
sl@0
   513
			if (CDebugBar* dbg = iScreen.DebugBar())
sl@0
   514
				{	
sl@0
   515
				RArray<TPtrC> debugText;
sl@0
   516
				dbg->DebugBarInfo(debugText);
sl@0
   517
				debugBar->DrawDebugBar(debugText.Array());
sl@0
   518
				debugText.Close();
sl@0
   519
				}
sl@0
   520
			}
sl@0
   521
#endif
sl@0
   522
		
sl@0
   523
		iScheduledWindowList = NULL;
sl@0
   524
		CWsMemoryManager::Static()->DisableReserve();
sl@0
   525
sl@0
   526
#ifdef USE_DEBUG_FRAME_CAPTURE
sl@0
   527
		if (iFrameCapture)
sl@0
   528
			{
sl@0
   529
			CaptureFrame(iAnimationRegion);
sl@0
   530
			}
sl@0
   531
#endif
sl@0
   532
		
sl@0
   533
		// DEBUG_REGION does not work with render stages.
sl@0
   534
		// These comments refer to what used to happen.
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
		
sl@0
   541
		//iScreen.Update();
sl@0
   542
		
sl@0
   543
		// At this point, if the DEBUG_REGION is being used, there should be no regions visible
sl@0
   544
		// of any colour.  If we see green, then it means an area of the screen was drawn to which
sl@0
   545
		// wasn't invalid, or the screen update call failed.  The former is more likely.
sl@0
   546
		// If we still see red or yellow it is a region that is not yet ready to draw.
sl@0
   547
sl@0
   548
		const TRect* rect = iAnimationRegion->RectangleList();
sl@0
   549
		TInt pixels = 0;
sl@0
   550
		for(TInt r = iAnimationRegion->Count(); r>0; r--)
sl@0
   551
			{
sl@0
   552
			pixels += (rect->Width()*rect->Height());
sl@0
   553
			rect++;
sl@0
   554
			}
sl@0
   555
		CWsActiveScheduler::Static()->StopDraw(pixels);
sl@0
   556
		
sl@0
   557
		TWindowServerEvent::NotifyScreenDrawingEvent(iAnimationRegion);
sl@0
   558
sl@0
   559
		iAnimationRegion = NULL; //iAnimationRegion must be reset to NULL before the call stack unwinds.
sl@0
   560
		iInvalid.Reset();
sl@0
   561
		animationRegion.Reset();
sl@0
   562
		}
sl@0
   563
	else
sl@0
   564
		{
sl@0
   565
		// There was nothing to compose. Signal that composition is complete.
sl@0
   566
		if (aFinished) 
sl@0
   567
			{
sl@0
   568
			*aFinished = KRequestPending;
sl@0
   569
			User::RequestComplete(aFinished, KErrNone);
sl@0
   570
			}
sl@0
   571
		
sl@0
   572
		CWsActiveScheduler::Static()->CancelPrepare();
sl@0
   573
		}
sl@0
   574
	
sl@0
   575
	floatingSpriteDirtyRegion.Reset();
sl@0
   576
	iInvalid.Reset();
sl@0
   577
	iTopElement.Reset();
sl@0
   578
	iAnimating = EFalse;
sl@0
   579
	
sl@0
   580
	if (futureAnimationRequired && iTimedDrawRect.Count() > 0 && !iScheduled)
sl@0
   581
		{
sl@0
   582
		// If this flag is set then it means there were already some animations scheduled when we ran,
sl@0
   583
		// but they themselves didn't run.  We need to make sure we have _something_ scheduled.
sl@0
   584
		CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen, iTimedDrawRect[0].iTime);
sl@0
   585
		iScheduled = ETrue;
sl@0
   586
		}
sl@0
   587
	
sl@0
   588
	if(iObserver)
sl@0
   589
		{
sl@0
   590
		iObserver->ScreenUpdated(iScreen.ScreenNumber());
sl@0
   591
		iObserver=NULL; //once signalled we are never going to call it again
sl@0
   592
		}
sl@0
   593
	LOG_SCREEN_REDRAW_END
sl@0
   594
	}
sl@0
   595
sl@0
   596
void CScreenRedraw::AddQuickFadeableRegions(TRegion& aRegion)
sl@0
   597
	{
sl@0
   598
	if (iQuickFadeList.Count() > 0)
sl@0
   599
		{
sl@0
   600
		for (TInt idx = iQuickFadeList.Count() - 1; idx >= 0; idx --)
sl@0
   601
			{
sl@0
   602
			CWsWindow* win = iQuickFadeList[ idx ];
sl@0
   603
sl@0
   604
			if ( !win->IsDSAHost() )
sl@0
   605
				{
sl@0
   606
				aRegion.Union(win->QuickFadeRegion() );
sl@0
   607
				}
sl@0
   608
			}
sl@0
   609
		aRegion.Tidy();
sl@0
   610
		}
sl@0
   611
	}
sl@0
   612
sl@0
   613
void CScreenRedraw::DoQuickFade(MWsGraphicsContext* aGc, TRegion& aAccumulatedDrawing)
sl@0
   614
	{
sl@0
   615
	if ( iQuickFadeList.Count() > 0 )
sl@0
   616
		{
sl@0
   617
		for ( TInt idx = iQuickFadeList.Count() - 1; idx >= 0 ; idx -- )
sl@0
   618
			{
sl@0
   619
sl@0
   620
			CWsWindow* win = iQuickFadeList[ idx ];
sl@0
   621
			
sl@0
   622
			if ( !win->IsDSAHost() )
sl@0
   623
				{				
sl@0
   624
				STACK_REGION winFadeRgn;				
sl@0
   625
				winFadeRgn.Copy( win->QuickFadeRegion() );
sl@0
   626
				winFadeRgn.SubRegion( aAccumulatedDrawing );
sl@0
   627
				winFadeRgn.Tidy();
sl@0
   628
				if ( !winFadeRgn.IsEmpty() && !winFadeRgn.CheckError() )
sl@0
   629
					{ // now fade any region that has not been redrawn (via win->Render())
sl@0
   630
					win->Fade( aGc, winFadeRgn );
sl@0
   631
					aAccumulatedDrawing.Union( winFadeRgn );	      		
sl@0
   632
					}		      	
sl@0
   633
	  			winFadeRgn.Close();
sl@0
   634
				}
sl@0
   635
			}
sl@0
   636
	
sl@0
   637
		iQuickFadeList.Reset();
sl@0
   638
		}
sl@0
   639
	}
sl@0
   640
sl@0
   641
/**
sl@0
   642
This function will iterate the timed rectangles and find any animation or sprite 
sl@0
   643
that are due to be scheduled for render now. Then schedule them and remove them 
sl@0
   644
from iTimedDrawRect.
sl@0
   645
@return ETrue if iTimedDrawRect still contains future animations
sl@0
   646
*/
sl@0
   647
TBool CScreenRedraw::ScheduleTimedRects(TRegion& aScheduledFloatingSpriteRegion)
sl@0
   648
 	{
sl@0
   649
	iNow.UniversalTime();
sl@0
   650
	TInt count(iTimedDrawRect.Count());
sl@0
   651
	TBool futureAnimationRequired = EFalse;
sl@0
   652
	while (0 < count)
sl@0
   653
		{
sl@0
   654
		if(iTimedDrawRect[0].iTime.Int64() <= iNow.Int64())
sl@0
   655
			{
sl@0
   656
			if (iScreen.ChangeTracking())
sl@0
   657
				{
sl@0
   658
				switch(iTimedDrawRect[0].iType)
sl@0
   659
					{
sl@0
   660
					case ECrpAnim:
sl@0
   661
						{
sl@0
   662
						CWsWindow* win = iTimedDrawRect[0].iWindow;
sl@0
   663
						WS_ASSERT_DEBUG(win,EWsPanicWindowNull);
sl@0
   664
						if(!win->IsTrackingVisibleRegion() || (win->IsTrackingVisibleRegion() && !win->VisibleRegion().IsEmpty()))
sl@0
   665
							{
sl@0
   666
							TRegionFix<1> region;
sl@0
   667
							region.AddRect(iTimedDrawRect[0].iRect);
sl@0
   668
							win->AddDirtyWindowRegion(region);
sl@0
   669
							ScheduleWindow(win);
sl@0
   670
							}
sl@0
   671
						break;
sl@0
   672
						}
sl@0
   673
					case EWindowAnim:
sl@0
   674
					case ESpriteAnim:
sl@0
   675
					case ETextCursor:
sl@0
   676
					case EWindowSprite:
sl@0
   677
						{
sl@0
   678
						CWsWindow* win = iTimedDrawRect[0].iWindow;
sl@0
   679
						WS_ASSERT_DEBUG(win,EWsPanicWindowNull);
sl@0
   680
						if(!win->IsTrackingVisibleRegion() || (win->IsTrackingVisibleRegion() && !win->VisibleRegion().IsEmpty()))
sl@0
   681
							{
sl@0
   682
							TRegionFix<1> region;
sl@0
   683
							region.AddRect(iTimedDrawRect[0].iRect);
sl@0
   684
							win->AddDirtySpriteRegion(region);
sl@0
   685
							ScheduleWindow(win);
sl@0
   686
							}
sl@0
   687
						break;
sl@0
   688
						}
sl@0
   689
					case EFloatingSprite:
sl@0
   690
					case EFloatingSpriteAnim:
sl@0
   691
						{
sl@0
   692
						aScheduledFloatingSpriteRegion.AddRect(iTimedDrawRect[0].iRect);
sl@0
   693
						break;
sl@0
   694
						}
sl@0
   695
					}
sl@0
   696
				
sl@0
   697
				TRect screenRect(iTimedDrawRect[0].iRect);
sl@0
   698
				if(iTimedDrawRect[0].iWindow)
sl@0
   699
					{
sl@0
   700
					screenRect.Move(iTimedDrawRect[0].iWindow->Origin()); // convert to screen coordinates
sl@0
   701
					}	
sl@0
   702
				}
sl@0
   703
			else
sl@0
   704
				{ 
sl@0
   705
				// schedule a screen dirty rect
sl@0
   706
				iInvalid.AddRect(iTimedDrawRect[0].iRect);
sl@0
   707
				}
sl@0
   708
			iTimedDrawRect.Remove(0);
sl@0
   709
			count--;
sl@0
   710
			}
sl@0
   711
		else
sl@0
   712
			{
sl@0
   713
			futureAnimationRequired = ETrue;	
sl@0
   714
			break;
sl@0
   715
			}
sl@0
   716
		}
sl@0
   717
	return futureAnimationRequired;
sl@0
   718
 	}
sl@0
   719
sl@0
   720
void CScreenRedraw::Render(CWsWindow& aWin, MWsGraphicsContext& aGc, const TWalkWindowTreeSchedule& aScheduler)
sl@0
   721
	{
sl@0
   722
	const TRegion* windowRegion = &(aScheduler.WindowRegion(aWin));
sl@0
   723
	const TRegion* spriteRegion = &(aScheduler.SpriteRegion(aWin));
sl@0
   724
sl@0
   725
	if(windowRegion->IsEmpty() && spriteRegion->IsEmpty()) 
sl@0
   726
	    return; //Don't call CWsWindow::Render if there is nothing that can be rendered 
sl@0
   727
	
sl@0
   728
	//Make sure we don't try to draw outside screen
sl@0
   729
	STACK_REGION clippedWindowRegion;
sl@0
   730
	if(!windowRegion->IsContainedBy(iScreen.RootWindow()->Abs()))
sl@0
   731
		{
sl@0
   732
		TRegionFix<1> screen(iScreen.RootWindow()->Abs());
sl@0
   733
		clippedWindowRegion.Intersection(*windowRegion, screen);
sl@0
   734
		windowRegion = &clippedWindowRegion;
sl@0
   735
		}
sl@0
   736
	STACK_REGION clippedSpriteRegion;
sl@0
   737
	if(!spriteRegion->IsContainedBy(iScreen.RootWindow()->Abs()))
sl@0
   738
		{
sl@0
   739
		TRegionFix<1> screen(iScreen.RootWindow()->Abs());
sl@0
   740
		clippedSpriteRegion.Intersection(*spriteRegion, screen);
sl@0
   741
		spriteRegion = &clippedSpriteRegion;
sl@0
   742
		}	
sl@0
   743
	
sl@0
   744
	if(!windowRegion->CheckError() && !spriteRegion->CheckError())
sl@0
   745
		{
sl@0
   746
		// Purple regions are about to be drawn
sl@0
   747
		DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80), windowRegion);
sl@0
   748
		DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80), spriteRegion);
sl@0
   749
		
sl@0
   750
		aWin.Render(aGc, *windowRegion, *spriteRegion); //When not in ChangeTracking mode, both windowRegion and spriteRegion points to the same region
sl@0
   751
		aWin.ClearScheduledRegion();
sl@0
   752
		aWin.ClearScheduledSpriteRegion();
sl@0
   753
		
sl@0
   754
		// Green regions have been drawn
sl@0
   755
		DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80), windowRegion);
sl@0
   756
		DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80), spriteRegion);
sl@0
   757
		}
sl@0
   758
	else
sl@0
   759
		{
sl@0
   760
		OomRender(aWin, aGc, aScheduler);		
sl@0
   761
		}
sl@0
   762
	
sl@0
   763
	clippedSpriteRegion.Close();
sl@0
   764
	clippedWindowRegion.Close();
sl@0
   765
	}
sl@0
   766
sl@0
   767
void CScreenRedraw::OomRender(CWsWindow& aWin, MWsGraphicsContext& aGc, const TWalkWindowTreeSchedule& aScheduler)
sl@0
   768
	{	
sl@0
   769
	const TRegion* windowRegion = &(aScheduler.WindowRegion(aWin));
sl@0
   770
	const TRegion* spriteRegion = &(aScheduler.SpriteRegion(aWin));
sl@0
   771
sl@0
   772
	WS_ASSERT_DEBUG(!(windowRegion->IsEmpty() && spriteRegion->IsEmpty()), EWsPanicRegionNull);
sl@0
   773
	
sl@0
   774
	TRect validWindow(aWin.Abs());
sl@0
   775
	validWindow.Intersection(iScreen.RootWindow()->Abs());
sl@0
   776
	
sl@0
   777
    TRegionFix<1> fallbackRegion(validWindow);
sl@0
   778
    
sl@0
   779
    if(windowRegion->CheckError())
sl@0
   780
        windowRegion = &fallbackRegion;
sl@0
   781
    if(spriteRegion->CheckError())
sl@0
   782
        spriteRegion = &fallbackRegion;
sl@0
   783
    
sl@0
   784
    // Purple regions are about to be drawn
sl@0
   785
    DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80), windowRegion);
sl@0
   786
    DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80), spriteRegion);
sl@0
   787
    
sl@0
   788
    aWin.Render(aGc, *windowRegion, *spriteRegion); //When not in ChangeTracking mode, both windowRegion and spriteRegion points to the same region
sl@0
   789
    aWin.ClearScheduledRegion();
sl@0
   790
    aWin.ClearScheduledSpriteRegion();
sl@0
   791
    
sl@0
   792
    // Green regions have been drawn
sl@0
   793
    DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80), windowRegion);
sl@0
   794
    DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80), spriteRegion);	
sl@0
   795
	}
sl@0
   796
sl@0
   797
//
sl@0
   798
void CScreenRedraw::DiscardAllSchedules()
sl@0
   799
	{
sl@0
   800
	ASSERT(!iAnimating);
sl@0
   801
	
sl@0
   802
	iTimedDrawRect.Reset();
sl@0
   803
	iInvalid.Reset();
sl@0
   804
	}
sl@0
   805
sl@0
   806
/**
sl@0
   807
Indicates that a window has moved or changed ordinal position so that the visible regions
sl@0
   808
of all windows needs to be recalculated
sl@0
   809
*/
sl@0
   810
void CScreenRedraw::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
sl@0
   811
	{
sl@0
   812
	if (iScreen.ChangeTracking() && iScreen.WindowVisibilityNotifier())
sl@0
   813
		return;
sl@0
   814
sl@0
   815
	iRegionUpdateScheduled = ETrue;
sl@0
   816
	ScheduleRedraw();
sl@0
   817
	if(!iScreen.ChangeTracking() && aDefinitelyDirty)
sl@0
   818
		{
sl@0
   819
		iInvalid.Union(*aDefinitelyDirty);
sl@0
   820
		// Cyan regions for invalidations caused by this:
sl@0
   821
		DEBUG_REGION(TRgb(0x00, 0xFF, 0xFF),TRgb(0x00, 0xFF, 0xFF, 0x20),aDefinitelyDirty);
sl@0
   822
		}
sl@0
   823
	}
sl@0
   824
sl@0
   825
/**
sl@0
   826
Only used in CHANGETRACKING mode.
sl@0
   827
Windows are scheduled to:
sl@0
   828
- if required, ask client to validate (from lower loop)
sl@0
   829
- render the dirty window region
sl@0
   830
*/
sl@0
   831
void CScreenRedraw::ScheduleWindow(CWsWindow* aWindow)
sl@0
   832
	{
sl@0
   833
	// Add a scheduled window to the linked list, ignoring duplicates
sl@0
   834
	CWsWindow* win = iScheduledWindowList;
sl@0
   835
	while (win && win != aWindow)
sl@0
   836
		{
sl@0
   837
		win = win->NextScheduled();
sl@0
   838
		}
sl@0
   839
	if (!win)
sl@0
   840
		{
sl@0
   841
		aWindow->SetNextScheduled(iScheduledWindowList);
sl@0
   842
		iScheduledWindowList = aWindow;
sl@0
   843
		}
sl@0
   844
	ScheduleRedraw();
sl@0
   845
	}
sl@0
   846
sl@0
   847
void CScreenRedraw::RemoveFromScheduledList(CWsWindow* aWindow)
sl@0
   848
	{
sl@0
   849
	// Search for the window
sl@0
   850
	CWsWindow* win = iScheduledWindowList;
sl@0
   851
	CWsWindow* previous = NULL;
sl@0
   852
	while (win && win != aWindow)
sl@0
   853
		{
sl@0
   854
		previous = win;
sl@0
   855
		win = win->NextScheduled();
sl@0
   856
		}
sl@0
   857
	if (win)
sl@0
   858
		{
sl@0
   859
		// Found it, remove from list
sl@0
   860
		if (win == iScheduledWindowList)
sl@0
   861
			{
sl@0
   862
			iScheduledWindowList = win->NextScheduled();
sl@0
   863
			}
sl@0
   864
		else
sl@0
   865
			{
sl@0
   866
			previous->SetNextScheduled(win->NextScheduled());
sl@0
   867
			}
sl@0
   868
		}
sl@0
   869
	}
sl@0
   870
sl@0
   871
void CScreenRedraw::RemoveFromTimedDrawList(CWsWindow* aWindow)
sl@0
   872
	{
sl@0
   873
	if(aWindow)
sl@0
   874
		{	
sl@0
   875
		TInt count(iTimedDrawRect.Count());
sl@0
   876
		TInt index = 0;
sl@0
   877
		while(index < count)
sl@0
   878
			{
sl@0
   879
			if (iTimedDrawRect[index].iWindow == aWindow)
sl@0
   880
				{
sl@0
   881
				iTimedDrawRect.Remove(index);
sl@0
   882
				--count;
sl@0
   883
				}
sl@0
   884
			else
sl@0
   885
				{
sl@0
   886
				++index;
sl@0
   887
				}
sl@0
   888
			}
sl@0
   889
		}
sl@0
   890
	}
sl@0
   891
sl@0
   892
/**
sl@0
   893
Recalculates visible regions and schedules redraws or queues redraw events in response to
sl@0
   894
any changes
sl@0
   895
*/
sl@0
   896
void CScreenRedraw::RegionUpdate()
sl@0
   897
	{
sl@0
   898
	TBool somethingScheduled = EFalse;
sl@0
   899
sl@0
   900
	if (iVisibilityUpdateScheduled)
sl@0
   901
		{
sl@0
   902
		iVisibilityUpdateScheduled = EFalse;
sl@0
   903
		somethingScheduled = ETrue;
sl@0
   904
		MWsWindowVisibilityNotifier* const notifier = iScreen.WindowVisibilityNotifier();
sl@0
   905
		if(notifier)
sl@0
   906
			{
sl@0
   907
			notifier->SendVisibilityChanges(); // Should result in one callback to SetWindowVisibility for each window subscribing for visible region updates
sl@0
   908
			}
sl@0
   909
		}
sl@0
   910
sl@0
   911
	if (iRegionUpdateScheduled)
sl@0
   912
		{
sl@0
   913
		iRegionUpdateScheduled = EFalse;
sl@0
   914
		somethingScheduled = ETrue;
sl@0
   915
		TWalkWindowTreeUpdateRegions updater(iScreen);
sl@0
   916
		updater.Walk();
sl@0
   917
		
sl@0
   918
		for ( TInt idx = iQuickFadeList.Count() - 1; idx >= 0; idx-- )
sl@0
   919
			{
sl@0
   920
			CWsWindow* win = iQuickFadeList[ idx ];
sl@0
   921
			const TRegion& quickFadeRegion = win->QuickFadeRegion();
sl@0
   922
			//If QuickFadeRegion().IsEmpty() we should remove the window from iQuickFadeList.
sl@0
   923
			//And if this window has not been drawn to the screen, then it is not possible to quick fade it.
sl@0
   924
			if (quickFadeRegion.IsEmpty() || !win->HasBeenDrawnToScreen())
sl@0
   925
				{
sl@0
   926
				iQuickFadeList.Remove( idx );
sl@0
   927
				}
sl@0
   928
			}
sl@0
   929
		}
sl@0
   930
sl@0
   931
	if (somethingScheduled)
sl@0
   932
		{
sl@0
   933
		TWsPointer::ReLogPointersCurrentWindows();
sl@0
   934
		CWsTop::TriggerRedraws(iScreen.RootWindow());
sl@0
   935
		}
sl@0
   936
	}
sl@0
   937
sl@0
   938
/**
sl@0
   939
To be called by MWsWindowVisibilityNotifier (when running in ChangeTracking mode) 
sl@0
   940
in response to MWsWindowVisibilityNotifier::SendVisibilityChanges()  
sl@0
   941
*/
sl@0
   942
void CScreenRedraw::SetWindowVisibility(const MWsWindow& aWindow, const TRegion& aVisibleRegion)
sl@0
   943
	{
sl@0
   944
	CWsWindow& win = static_cast<CWsWindow&>(const_cast<MWsWindow&>(aWindow));
sl@0
   945
	WS_ASSERT_DEBUG(win.IsTrackingVisibleRegion(), EWsPanicVisibleRegionTracking);
sl@0
   946
	WS_ASSERT_DEBUG(!aVisibleRegion.CheckError(), EWsPanicRegion);
sl@0
   947
	
sl@0
   948
	if(aVisibleRegion.IsEmpty() && !win.VisibleRegion().IsEmpty())
sl@0
   949
		{
sl@0
   950
		win.ClearVisibleRegion();
sl@0
   951
		}
sl@0
   952
	else if(!aVisibleRegion.IsEmpty() && !aVisibleRegion.CheckError())
sl@0
   953
		{
sl@0
   954
		// Assert that aVisibleRegion is contained by aWin
sl@0
   955
		TRect bounds = win.AbsRect();
sl@0
   956
		bounds.Resize(1,1);
sl@0
   957
		WS_ASSERT_DEBUG(bounds.Contains(aVisibleRegion.BoundingRect().iTl), EWsPanicRegion);
sl@0
   958
		WS_ASSERT_DEBUG(bounds.Contains(aVisibleRegion.BoundingRect().iBr), EWsPanicRegion);
sl@0
   959
		win.SetVisibleRegion(aVisibleRegion, NULL);
sl@0
   960
		}
sl@0
   961
	else if(aVisibleRegion.CheckError())
sl@0
   962
		{
sl@0
   963
		const TRegionFix<1> fallback(win.Abs());
sl@0
   964
		win.SetVisibleRegion(fallback, NULL);
sl@0
   965
		}
sl@0
   966
	}
sl@0
   967
sl@0
   968
/**
sl@0
   969
To be called by MWsWindowVisibilityNotifier (when running in ChangeTracking mode) when the 
sl@0
   970
iVisibleRegion is changed for a window that subscribes for this information. The visible region 
sl@0
   971
of the window has not been updated yet at the time of this function call, but the new metrics will be retrieved 
sl@0
   972
from MWsWindowVisibilityNotifier (through a call to MWsWindowVisibilityObserver::SetWindowVisibility())
sl@0
   973
and set to each window next time OnAnimation is called.   
sl@0
   974
*/
sl@0
   975
void CScreenRedraw::VisibilityChanged()
sl@0
   976
	{
sl@0
   977
	iVisibilityUpdateScheduled = ETrue;
sl@0
   978
	ScheduleRedraw();
sl@0
   979
	}
sl@0
   980
sl@0
   981
void CScreenRedraw::SetObserver(MWsScreenRedrawObserver* aObserver)
sl@0
   982
	{
sl@0
   983
	iObserver = aObserver;
sl@0
   984
	}
sl@0
   985
sl@0
   986
/**
sl@0
   987
Returns ETrue if an update or animation is scheduled.
sl@0
   988
sl@0
   989
Note: Now that WSERV surface and composition updates are asynchronous (with NGA)
sl@0
   990
it is possible for this method to return EFalse, even if the last update has
sl@0
   991
not been completed by the rendering pipeline. This is dependant on the configuration
sl@0
   992
and implementation of render stages that make up the rendering pipeline.
sl@0
   993
sl@0
   994
If in doubt, it is best to call CScreenRedraw::DoRedrawNow(), which will not return
sl@0
   995
untill all updates have been signalled as being complete (note that if signalling
sl@0
   996
is not used, then even this cannot guarantee completion).
sl@0
   997
*/ 
sl@0
   998
TBool CScreenRedraw::IsUpdatePending()
sl@0
   999
	{
sl@0
  1000
	if(iScheduled || iAnimating)
sl@0
  1001
		return ETrue;
sl@0
  1002
	else 
sl@0
  1003
		return EFalse;
sl@0
  1004
	}
sl@0
  1005
sl@0
  1006
/**
sl@0
  1007
 Overidding MWsObjectProvider
sl@0
  1008
*/
sl@0
  1009
TAny* CScreenRedraw::ResolveObjectInterface(TUint aTypeId)
sl@0
  1010
	{
sl@0
  1011
	TAny* interface = NULL;
sl@0
  1012
sl@0
  1013
	switch (aTypeId)
sl@0
  1014
		{
sl@0
  1015
		case KWsScreenRedraw:
sl@0
  1016
			interface = static_cast<MWsScreenRedraw*>(this);
sl@0
  1017
			break;
sl@0
  1018
		}
sl@0
  1019
sl@0
  1020
	if (!interface)
sl@0
  1021
		{
sl@0
  1022
		interface = iRenderStages->ResolveObjectInterface(aTypeId);
sl@0
  1023
		}
sl@0
  1024
	
sl@0
  1025
	return interface;
sl@0
  1026
	}
sl@0
  1027
sl@0
  1028
const TRegion * CScreenRedraw::AnimationRegion() const
sl@0
  1029
	{
sl@0
  1030
	if (iAnimating)
sl@0
  1031
		return iAnimationRegion;
sl@0
  1032
	else
sl@0
  1033
		return 0;
sl@0
  1034
	}
sl@0
  1035
sl@0
  1036
void CScreenRedraw::UpdateDevice()
sl@0
  1037
	{
sl@0
  1038
	//this used to call iScreen->Update, not needed anymore in NGA
sl@0
  1039
	}
sl@0
  1040
sl@0
  1041
TBool CScreenRedraw::IsQuickFadeScheduled( CWsWindow* aWin ) const
sl@0
  1042
	{
sl@0
  1043
	const TInt idx = iQuickFadeList.Find( aWin );	
sl@0
  1044
	return (idx > KErrNotFound);
sl@0
  1045
	}
sl@0
  1046
sl@0
  1047
void CScreenRedraw::ScheduleQuickFade( CWsWindow* aWin )
sl@0
  1048
	{
sl@0
  1049
	const TInt idx = iQuickFadeList.Find( aWin );	
sl@0
  1050
	if ( idx == KErrNotFound )
sl@0
  1051
		{ // not yet enlisted		
sl@0
  1052
		if ( KErrNone != iQuickFadeList.Append(aWin) )
sl@0
  1053
			{  // out of resources?
sl@0
  1054
			const TRegion& winReg = aWin->VisibleRegion();
sl@0
  1055
			ScheduleRegionUpdate( &winReg );
sl@0
  1056
			}		
sl@0
  1057
		}
sl@0
  1058
	}
sl@0
  1059
sl@0
  1060
void CScreenRedraw::RemoveFromQuickFadeList( CWsWindow* aWin )
sl@0
  1061
	{	
sl@0
  1062
	const TInt idx = iQuickFadeList.Find( aWin );	
sl@0
  1063
	if ( idx > KErrNotFound )
sl@0
  1064
		{
sl@0
  1065
		iQuickFadeList.Remove( idx );
sl@0
  1066
		}
sl@0
  1067
	}
sl@0
  1068
sl@0
  1069
void CScreenRedraw::AcceptFadeRequest( CWsWindow* aWin, TBool aFadeOn )
sl@0
  1070
	{
sl@0
  1071
	if (aFadeOn)
sl@0
  1072
		{
sl@0
  1073
		if (aWin->WinType() != EWinTypeGroup)
sl@0
  1074
			{
sl@0
  1075
			ScheduleQuickFade(aWin);
sl@0
  1076
			ScheduleRegionUpdate(NULL);
sl@0
  1077
			}
sl@0
  1078
		}
sl@0
  1079
	else
sl@0
  1080
		{ // fade off, just initiate redraw
sl@0
  1081
		if ( !aWin->IsDSAHost() )
sl@0
  1082
			{
sl@0
  1083
			AddRedrawRegion(aWin->VisibleRegion() );
sl@0
  1084
			ScheduleRegionUpdate(NULL);
sl@0
  1085
			}
sl@0
  1086
		}
sl@0
  1087
	}
sl@0
  1088
sl@0
  1089
sl@0
  1090
#ifdef USE_DEBUG_FRAME_CAPTURE
sl@0
  1091
class TTruncateOverflow : public TDesOverflow
sl@0
  1092
	{
sl@0
  1093
public: //from TDesOverflow
sl@0
  1094
	void Overflow(TDes&) {};
sl@0
  1095
	};
sl@0
  1096
sl@0
  1097
void CScreenRedraw::CaptureFrame(const TRegion* aRegion)
sl@0
  1098
	{
sl@0
  1099
	MWsScreenDevice* screenDevice = static_cast<MWsScreen&>(iScreen).ObjectInterface<MWsScreenDevice>();
sl@0
  1100
	WS_ASSERT_ALWAYS(screenDevice, EWsPanicScreenDeviceMissing);
sl@0
  1101
	const TSize screenSize(screenDevice->SizeInPixels());
sl@0
  1102
	const TDisplayMode screenDisplayMode(screenDevice->DisplayMode());
sl@0
  1103
sl@0
  1104
	//copy the whole screen
sl@0
  1105
	TRAPD(err, SetupFrameCaptureResourcesL(screenSize, screenDisplayMode);
sl@0
  1106
	           screenDevice->CopyScreenToBitmapL(iDebugBitmap, TRect(screenSize)));
sl@0
  1107
sl@0
  1108
	if(err)
sl@0
  1109
		{
sl@0
  1110
		if(wsDebugLog)
sl@0
  1111
			wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, _L("CScreenRedraw::CaptureFrame(): Failed to create resources for screen capture"));
sl@0
  1112
		return;
sl@0
  1113
		}
sl@0
  1114
sl@0
  1115
	//remove what's not part of the region
sl@0
  1116
	iDebugBitmapGc->Activate(iDebugBitmapDevice);
sl@0
  1117
	
sl@0
  1118
	if(aRegion && !aRegion->IsEmpty() && !aRegion->CheckError())
sl@0
  1119
		{
sl@0
  1120
		RWsRegion inverseRegion;
sl@0
  1121
		inverseRegion.AddRect(TRect(screenSize));
sl@0
  1122
		const TRect* rectList = aRegion->RectangleList();
sl@0
  1123
		for(TInt i = aRegion->Count() - 1; i >= 0; i--)
sl@0
  1124
			{
sl@0
  1125
			inverseRegion.SubRect(rectList[i]);
sl@0
  1126
			}
sl@0
  1127
		if(!inverseRegion.CheckError())
sl@0
  1128
			{
sl@0
  1129
			iDebugBitmapGc->SetDrawMode(CGraphicsContext::EDrawModeWriteAlpha);
sl@0
  1130
			iDebugBitmapGc->SetClippingRegion(inverseRegion);
sl@0
  1131
			iDebugBitmapGc->SetBrushColor(TRgb(55, 55, 55, 0));
sl@0
  1132
			iDebugBitmapGc->Clear();
sl@0
  1133
			iDebugBitmapGc->CancelClippingRegion();
sl@0
  1134
			iDebugBitmapGc->SetDrawMode(CGraphicsContext::EDrawModePEN);
sl@0
  1135
			}
sl@0
  1136
		inverseRegion.Close();
sl@0
  1137
		}
sl@0
  1138
sl@0
  1139
	//save to file
sl@0
  1140
	const TUint timestamp = User::FastCounter();	
sl@0
  1141
	TFileName filename;
sl@0
  1142
	TTruncateOverflow overflow;
sl@0
  1143
	filename.AppendFormat(iFrameCaptureLocation, &overflow);
sl@0
  1144
	filename.AppendFormat(_L("frame_%010u.mbm"), &overflow, timestamp);
sl@0
  1145
	iDebugBitmap->Save(filename);
sl@0
  1146
sl@0
  1147
	//log region
sl@0
  1148
	LogRegion(filename, _L(" CScreenRedraw::CaptureFrame() "), aRegion);	
sl@0
  1149
	}
sl@0
  1150
sl@0
  1151
void CScreenRedraw::SetupFrameCaptureResourcesL(const TSize& aScreenSize, TDisplayMode aScreenDisplayMode)
sl@0
  1152
	{
sl@0
  1153
	//make sure the existing bitmap has the correct display mode
sl@0
  1154
	if(iDebugBitmap && iDebugBitmap->DisplayMode() != aScreenDisplayMode)
sl@0
  1155
		{
sl@0
  1156
		if(iDebugBitmap->SetDisplayMode(aScreenDisplayMode) != KErrNone)
sl@0
  1157
			{
sl@0
  1158
			delete iDebugBitmap;
sl@0
  1159
			iDebugBitmap = NULL;
sl@0
  1160
			delete iDebugBitmapDevice;
sl@0
  1161
			iDebugBitmapDevice = NULL;
sl@0
  1162
			}
sl@0
  1163
		}
sl@0
  1164
sl@0
  1165
	//make sure the existing bitmap has the correct size
sl@0
  1166
	if(iDebugBitmap && iDebugBitmap->SizeInPixels() != aScreenSize)
sl@0
  1167
		{
sl@0
  1168
		if(iDebugBitmap->Resize(aScreenSize) != KErrNone)
sl@0
  1169
			{
sl@0
  1170
			delete iDebugBitmap;
sl@0
  1171
			iDebugBitmap = NULL;
sl@0
  1172
			delete iDebugBitmapDevice;
sl@0
  1173
			iDebugBitmapDevice = NULL;
sl@0
  1174
			}
sl@0
  1175
		}
sl@0
  1176
sl@0
  1177
	//make sure the bitmap and bitmap device is created
sl@0
  1178
	if(!iDebugBitmap)
sl@0
  1179
		{
sl@0
  1180
		WS_ASSERT_ALWAYS(!iDebugBitmapDevice, EWsPanicTemp); //this should never occur, they should always be created/destroyed in tandem
sl@0
  1181
		CFbsBitmap* bitmap = new(ELeave) CFbsBitmap;
sl@0
  1182
		CleanupStack::PushL(bitmap);
sl@0
  1183
		User::LeaveIfError(bitmap->Create(aScreenSize, aScreenDisplayMode));
sl@0
  1184
sl@0
  1185
		iDebugBitmapDevice = CFbsBitmapDevice::NewL(bitmap);
sl@0
  1186
		iDebugBitmap = bitmap;
sl@0
  1187
		CleanupStack::Pop(bitmap);
sl@0
  1188
		}
sl@0
  1189
	
sl@0
  1190
	//make sure the gc is created
sl@0
  1191
	if(!iDebugBitmapGc)
sl@0
  1192
		{
sl@0
  1193
		User::LeaveIfError(iDebugBitmapDevice->CreateContext(iDebugBitmapGc));
sl@0
  1194
		}
sl@0
  1195
	}
sl@0
  1196
sl@0
  1197
void CScreenRedraw::LogRegion(const TDesC& aPrefix, const TDesC& aFunctionName, const TRegion* aRegion)
sl@0
  1198
	{
sl@0
  1199
	if(!wsDebugLog)
sl@0
  1200
		return;
sl@0
  1201
sl@0
  1202
	TBuf<LogTBufSize> log;
sl@0
  1203
	TTruncateOverflow overflow;
sl@0
  1204
	TInt rectCount = (aRegion == NULL ? 0 : aRegion->Count());
sl@0
  1205
	log.AppendFormat(aPrefix, &overflow);
sl@0
  1206
	log.AppendFormat(aFunctionName, &overflow);
sl@0
  1207
	log.AppendFormat(_L("region [%d,"), &overflow, rectCount);
sl@0
  1208
	if (rectCount > 0)
sl@0
  1209
		{
sl@0
  1210
		const TRect* rectangles = aRegion->RectangleList();
sl@0
  1211
		TBuf<1> comma;
sl@0
  1212
		for (TInt ii = 0; ii < rectCount; ii++)
sl@0
  1213
			{
sl@0
  1214
			TRect current = rectangles[ii];
sl@0
  1215
			log.AppendFormat(_L("%S{{%d,%d},{%d,%d}}"), &overflow, &comma,
sl@0
  1216
                             current.iTl.iX,current.iTl.iY,current.iBr.iX,current.iBr.iY);
sl@0
  1217
			comma = _L(",");
sl@0
  1218
			}
sl@0
  1219
		}
sl@0
  1220
	else
sl@0
  1221
		{
sl@0
  1222
		log.AppendFormat(_L("NULL"), &overflow);
sl@0
  1223
		}
sl@0
  1224
	log.AppendFormat(_L("]"), &overflow);
sl@0
  1225
	wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
sl@0
  1226
	}
sl@0
  1227
#endif //USE_DEBUG_FRAME_CAPTURE
sl@0
  1228