1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nonnga/SERVER/ScreenRedraw.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,733 @@
1.4 +// Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +// SCREEN_REDRAW.CPP
1.18 +//
1.19 +//
1.20 +
1.21 +#include "ScreenRedraw.h"
1.22 +
1.23 +#include <hal.h>
1.24 +
1.25 +#include "debugbar.h"
1.26 +#include "screen.h"
1.27 +#include "inifile.h"
1.28 +#include "offscreenbitmap.h"
1.29 +#include "wspluginmanager.h"
1.30 +#include "pointer.h"
1.31 +#include "rootwin.h"
1.32 +#include "walkwindowtree.h"
1.33 +#include "wstop.h"
1.34 +#include "WsMemMgr.h"
1.35 +#include "Graphics/WsRenderStageFactory.h"
1.36 +#include "Graphics/WsRenderStage.h"
1.37 +#include "EVENT.H"
1.38 +
1.39 +GLREF_D CDebugLogBase *wsDebugLog;
1.40 +
1.41 +#ifdef USE_DEBUG_REGIONS
1.42 +# define DEBUG_REGION(col,fill,reg) DebugRegion(col,fill,reg)
1.43 +# define DEBUG_RECT(col,fill,rect) DebugRect(col,fill,rect)
1.44 +#else
1.45 +# define DEBUG_REGION(col,fill,reg)
1.46 +# define DEBUG_RECT(col,fill,rect)
1.47 +#endif
1.48 +
1.49 +#if defined(__WINS__) && defined(_DEBUG)
1.50 +# define DEBUGOSB { CWsOffScreenBitmap * ofb = iScreen.OffScreenBitmap(); if (ofb) ofb->Update(); }
1.51 +#else
1.52 +# define DEBUGOSB
1.53 +#endif
1.54 +
1.55 +#ifdef _DEBUG
1.56 +# define LOG_SCREEN_REDRAW_START {if (wsDebugLog) {_LIT(KLogScreenRedrawStart, ">> CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawStart);}}
1.57 +# define LOG_SCREEN_REDRAW_END {if (wsDebugLog) {_LIT(KLogScreenRedrawEnd, "<< CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawEnd);}}
1.58 +#else
1.59 +# define LOG_SCREEN_REDRAW_START
1.60 +# define LOG_SCREEN_REDRAW_END
1.61 +#endif
1.62 +
1.63 +CScreenRedraw::TTimedRect::TTimedRect(const TRect& aRect, const TTime& aTime):
1.64 + iRect(aRect), iTime(aTime)
1.65 + {
1.66 + }
1.67 +
1.68 +TInt CScreenRedraw::TTimedRect::Compare(const TTimedRect& aOne,const TTimedRect& aOther)
1.69 + {
1.70 + if(aOne.iTime < aOther.iTime)
1.71 + return -1;
1.72 + else if(aOne.iTime > aOther.iTime)
1.73 + return 1;
1.74 + else
1.75 + return 0;
1.76 + }
1.77 +
1.78 +CScreenRedraw * CScreenRedraw::NewL(CScreen& aScreen)
1.79 + {
1.80 + CScreenRedraw * self = new (ELeave) CScreenRedraw(aScreen);
1.81 + CleanupStack::PushL(self);
1.82 + self->ConstructL();
1.83 + CleanupStack::Pop(self);
1.84 + return self;
1.85 + }
1.86 +
1.87 +CScreenRedraw::CScreenRedraw(CScreen& aScreen): iScreen(aScreen)
1.88 + {
1.89 + }
1.90 +
1.91 +CScreenRedraw::~CScreenRedraw()
1.92 + {
1.93 + CWsRenderStage * stage = iRenderStages;
1.94 + while (stage!=NULL)
1.95 + {
1.96 + CWsRenderStage *next=stage->Next();
1.97 + delete stage;
1.98 + stage=next;
1.99 + }
1.100 + iTimedDrawRect.Close();
1.101 + iInvalid.Close();
1.102 + iTopLayer.Close();
1.103 + iBannedRegion.Close();
1.104 + }
1.105 +
1.106 +void CScreenRedraw::ConstructL()
1.107 + {
1.108 + CWsPluginManager * pluginMgr = CWsTop::WindowServer()->PluginManager();
1.109 +
1.110 + // Setup the render stages for this screen:
1.111 + _LIT(KDefaultRenderStages, "std");
1.112 + _LIT(KDefaultFlickerFreeRenderStages, "flickerbuffer std");
1.113 + _LIT(KRenderStages,"RENDERSTAGES");
1.114 + TPtrC stagesString;
1.115 + const TBool customStages = WsIniFile->FindVar(iScreen.ScreenNumber(),KRenderStages,stagesString);
1.116 +
1.117 + // If noone specifies stages for this screen, assume the standard implementation:
1.118 + const TDesC * stages;
1.119 + if (customStages)
1.120 + stages = &stagesString;
1.121 + else if (iScreen.OffScreenBitmap())
1.122 + stages = &KDefaultFlickerFreeRenderStages();
1.123 + else
1.124 + stages = &KDefaultRenderStages();
1.125 +
1.126 + CWsRenderStage * lastStage = 0;
1.127 +
1.128 + // Parse the string for implementation IDs:
1.129 + TLex lex(*stages);
1.130 + while(true)
1.131 + {
1.132 + TPtrC ptr = lex.NextToken();
1.133 + if (ptr.Length() > 0)
1.134 + {
1.135 + TInt err = KErrNone;
1.136 + MWsRenderStageFactory * factory = pluginMgr->FindNamedImplementation<MWsRenderStageFactory>(ptr);
1.137 + if (factory)
1.138 + {
1.139 + CWsRenderStage * stage = 0;
1.140 + TRAP(err, stage = factory->CreateStageL(static_cast<MWsScreen*>(&iScreen), this));
1.141 + if (err == KErrNone)
1.142 + {
1.143 + if (!stage)
1.144 + {
1.145 + err = KErrNotFound;
1.146 + }
1.147 + else
1.148 + {
1.149 + if (lastStage)
1.150 + lastStage->SetNext(stage);
1.151 + else
1.152 + iRenderStages = stage;
1.153 + lastStage = stage;
1.154 + }
1.155 + }
1.156 + }
1.157 + else
1.158 + {
1.159 + err = KErrNotFound;
1.160 + }
1.161 +
1.162 + if (wsDebugLog)
1.163 + {
1.164 + TBuf<64> buf;
1.165 + if (err == KErrNone)
1.166 + {
1.167 + _LIT(KAddedRenderStage,"Added render stage: ");
1.168 + buf.Append(KAddedRenderStage);
1.169 + buf.Append(ptr);
1.170 + wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,buf);
1.171 + }
1.172 + else
1.173 + {
1.174 + _LIT(KMissingRenderStage,"Failed to add render stage (%d): ");
1.175 + buf.Append(KMissingRenderStage);
1.176 + buf.Append(ptr);
1.177 + wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,buf,err);
1.178 + }
1.179 + }
1.180 + }
1.181 + else
1.182 + {
1.183 + break;
1.184 + }
1.185 + }
1.186 + }
1.187 +
1.188 +const TTime& CScreenRedraw::Now() const
1.189 + {
1.190 + if(!iAnimating)
1.191 + {
1.192 + iNow.UniversalTime();
1.193 + }
1.194 + return iNow;
1.195 + }
1.196 +
1.197 +void CScreenRedraw::ScheduleRender(const TTimeIntervalMicroSeconds& aFromNow)
1.198 + {
1.199 + iRenderScheduled = ETrue;
1.200 + TTime then(Now() + aFromNow);
1.201 + if ((!iScheduled) || then < iNext)
1.202 + iNext = then;
1.203 + iScheduled = ETrue;
1.204 + CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
1.205 + }
1.206 +
1.207 +void CScreenRedraw::ScheduleRedraw()
1.208 + {
1.209 + iNext = Now();
1.210 + iScheduled = ETrue;
1.211 +
1.212 + // The other scheduler also removes future animations which this one encompasses.
1.213 + // We choose not to do the region calculations needed to achieve that here.
1.214 + MWsAnimationScheduler* animSched=CWsTop::WindowServer()->AnimationScheduler();
1.215 + if (animSched)
1.216 + {
1.217 + animSched->ScheduleRedraw(iScreen,iNext);
1.218 + }
1.219 + }
1.220 +
1.221 +void CScreenRedraw::ScheduleAnimation(const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow,const TTimeIntervalMicroSeconds& /*aFreq*/,const TTimeIntervalMicroSeconds& /*aStop*/)
1.222 + {
1.223 + TRect test(aRect);
1.224 + test.Intersection(iScreen.DrawableArea());
1.225 + if(!test.IsEmpty())
1.226 + {
1.227 + const TTime then(Now() + aFromNow);
1.228 + TTimedRect tRect(aRect, then);
1.229 +
1.230 + const TInt error = iTimedDrawRect.InsertInOrderAllowRepeats(tRect,TTimedRect::Compare);
1.231 + if (KErrNone == error)
1.232 + {
1.233 + if (iScheduled)
1.234 + {
1.235 + if (then < iNext)
1.236 + {
1.237 + iNext = then;
1.238 + }
1.239 + }
1.240 + else
1.241 + {
1.242 + iNext = then;
1.243 + iScheduled = ETrue;
1.244 + }
1.245 + // remove further futures that are completely contained
1.246 + TInt count = iTimedDrawRect.Count();
1.247 + for(TInt i=0; i<count; i++)
1.248 + {
1.249 + const TTimedRect& future = iTimedDrawRect[i];
1.250 + if(future.iTime.Int64() > then.Int64())
1.251 + {
1.252 + TRect rect(aRect);
1.253 + rect.BoundingRect(future.iRect);
1.254 + if(rect == aRect) // future is completely contained within aRect
1.255 + {
1.256 + iTimedDrawRect.Remove(i);
1.257 + count--;
1.258 + i--;
1.259 + }
1.260 + }
1.261 + }
1.262 + CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
1.263 +
1.264 + // Blue rectangles for scheduled animations
1.265 + DEBUG_RECT(TRgb(0x00, 0x00, 0xFF),TRgb(0x00, 0x00, 0xFF, 0x20),&aRect);
1.266 + }
1.267 + }
1.268 + }
1.269 +
1.270 +// This adds a region to the stored invalid region.
1.271 +// The invalid region is the area of the screen that needs to be redrawn in addition to any animations.
1.272 +// The draw region is the area of the screen on which only the top window needs to be redrawn.
1.273 +// If the top window has transparency, this can only be true when it has been made newly visible.
1.274 +// The value of aSchedule could be determined automatically from iAnimating, but passing it this way
1.275 +// allows us to have the assert, which is a very valuable assert.
1.276 +void CScreenRedraw::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
1.277 + {
1.278 + WS_ASSERT_DEBUG(!iAnimating || !aSchedule, EWsPanicScheduledRedraw);
1.279 +
1.280 + if(aRegion.CheckError())
1.281 + {
1.282 + iInvalid.ForceError();
1.283 +
1.284 + if (aSchedule)
1.285 + ScheduleRedraw();
1.286 + }
1.287 + else if(aRegion.Count()) // often called despite window not being visible
1.288 + {
1.289 + if (aDepth == ERedrawAll)
1.290 + {
1.291 + // red lines for an invalid region which is ready to be drawn
1.292 + DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x20),&aRegion);
1.293 +
1.294 + iInvalid.Union(aRegion);
1.295 +
1.296 + if (aSchedule)
1.297 + ScheduleRedraw();
1.298 + }
1.299 + else
1.300 + {
1.301 + // yellow lines for a valid region which we will draw on top of
1.302 + DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x20),&aRegion);
1.303 +
1.304 + iTopLayer.Union(aRegion);
1.305 +
1.306 + if (aSchedule)
1.307 + ScheduleRedraw();
1.308 + }
1.309 + }
1.310 + }
1.311 +
1.312 +// This causes any asynchronously scheduled redraw to happen immediately
1.313 +// It should be avoided where possible for performance reasons, but is
1.314 +// needed whenever the redraw store is discarded for a window which still
1.315 +// has a redraw region pending.
1.316 +void CScreenRedraw::DoRedrawNow()
1.317 + {
1.318 + if(!iAnimating)
1.319 + CWsTop::WindowServer()->AnimationScheduler()->DoRedrawNow(iScreen);
1.320 + }
1.321 +
1.322 +#ifdef USE_DEBUG_REGIONS
1.323 +void CScreenRedraw::DebugRect(TRgb aColor, TRgb aFill, const TRect* aRect)
1.324 + {
1.325 + if (aRect)
1.326 + {
1.327 + CFbsBitGc * gc = iScreen.GetBitGc();
1.328 + gc->SetPenColor(aColor);
1.329 + gc->SetPenStyle(CGraphicsContext::ESolidPen);
1.330 + gc->SetPenSize(TSize(2,2));
1.331 + gc->SetBrushColor(aFill);
1.332 + gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
1.333 + TRect smaller = *aRect;
1.334 + smaller.iBr.iX -= 1;
1.335 + smaller.iBr.iY -= 1;
1.336 + gc->DrawRect(smaller);
1.337 + iScreen.Update();
1.338 + }
1.339 + }
1.340 +
1.341 +void CScreenRedraw::DebugRegion(TRgb aColor, TRgb aFill, const TRegion * aRegion)
1.342 + {
1.343 + if (aRegion)
1.344 + {
1.345 + CFbsBitGc * gc = iScreen.GetBitGc();
1.346 + gc->SetPenColor(aColor);
1.347 + gc->SetPenStyle(CGraphicsContext::ESolidPen);
1.348 + gc->SetPenSize(TSize(2,2));
1.349 + gc->SetBrushColor(aFill);
1.350 + gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
1.351 + for (const TRect *rect = aRegion->RectangleList(); rect - aRegion->RectangleList() < aRegion->Count(); ++rect)
1.352 + {
1.353 + TRect smaller = *rect;
1.354 + smaller.iBr.iX -= 1;
1.355 + smaller.iBr.iY -= 1;
1.356 + gc->DrawRect(smaller);
1.357 + }
1.358 + iScreen.Update();
1.359 + }
1.360 + }
1.361 +#endif
1.362 +
1.363 +void CScreenRedraw::OnAnimation()
1.364 + {
1.365 + LOG_SCREEN_REDRAW_START
1.366 + ASSERT(!iAnimating);
1.367 + ASSERT(iScheduled);
1.368 + iAnimating = ETrue;
1.369 + iScheduled = EFalse;
1.370 + TBool futureAnimationRequired = EFalse;
1.371 +
1.372 + CWsActiveScheduler::Static()->PrepareDraw();
1.373 +
1.374 + // Calculate any updates required by region changes:
1.375 + RegionUpdate();
1.376 +
1.377 + // Add the timed rectangles to the invalid region:
1.378 + iNow.UniversalTime();
1.379 + TInt count(iTimedDrawRect.Count());
1.380 + while (0 < count)
1.381 + {
1.382 + if(iTimedDrawRect[0].iTime.Int64() <= iNow.Int64())
1.383 + {
1.384 + iInvalid.AddRect(iTimedDrawRect[0].iRect);
1.385 + iTimedDrawRect.Remove(0);
1.386 + count--;
1.387 + }
1.388 + else
1.389 + {
1.390 + futureAnimationRequired = ETrue;
1.391 + break;
1.392 + }
1.393 + }
1.394 +
1.395 + // Animating rectangles could cause iInvalid to overlap iTopLayer,
1.396 + // in which case iTopLayer won't work.
1.397 + iTopLayer.SubRegion(iInvalid);
1.398 + iTopLayer.Intersect(iScreen.RootWindow()->WindowArea());
1.399 + iTopLayer.Tidy();
1.400 + // Anything in the top layer is implcitly invalid
1.401 + iInvalid.Union(iTopLayer);
1.402 + iInvalid.Intersect(iScreen.RootWindow()->WindowArea());
1.403 +
1.404 +/*
1.405 + if (const CDebugBar* dbg = iScreen.DebugBar())
1.406 + {
1.407 + iTopLayer.SubRect(dbg->Rect());
1.408 + iInvalid.SubRect(dbg->Rect());
1.409 + }
1.410 +*/
1.411 +
1.412 + iInvalid.Tidy();
1.413 +
1.414 + if(iInvalid.CheckError()) //Will: agree with Andy, want bounding rects instead!
1.415 + {
1.416 + iTopLayer.Clear();
1.417 + iInvalid.Clear();
1.418 + iInvalid.Copy(iScreen.RootWindow()->WindowArea()); // assumed cannot fail, all regions can contain at least 1 rect..
1.419 + }
1.420 +
1.421 + iInvalid.SubRegion( iBannedRegion );
1.422 + iInvalid.Tidy();
1.423 + iTopLayer.SubRegion( iBannedRegion );
1.424 + iTopLayer.Tidy();
1.425 +
1.426 + STACK_REGION invalidCopy;
1.427 + invalidCopy.Copy(iInvalid);
1.428 + TWalkWindowTreeScheduleRegions regionScheduler(&invalidCopy, iTopLayer);
1.429 + TWalkWindowTreeScheduleFallback fallbackScheduler(iScreen.FallbackMap());
1.430 + TWalkWindowTreeSchedule * scheduler = ®ionScheduler;
1.431 +
1.432 + // At this point, if the DEBUG_REGION is being used:
1.433 + // Red represents invalid regions that need to be redrawn completely.
1.434 + // Yellow represents regions that only need the top window to be drawn.
1.435 + // Blue represents regions which are being animated server side.
1.436 + if (iRenderScheduled || !iInvalid.IsEmpty())
1.437 + {
1.438 + iRenderScheduled = EFalse;
1.439 + // invalidCopy.ForceError(); //### DEBUG
1.440 +
1.441 + iScreen.RootWindow()->WalkWindowTree(regionScheduler,EWalkChildren);
1.442 + if (!regionScheduler.ScheduledRegionsOk())
1.443 + {
1.444 + // our region calculations for what to draw failed at some point.
1.445 + // From this point on we MUST NOT rely on allocating memory
1.446 + // Andy - localRedrawRegion allocates
1.447 + // Andy - setPenSize allocates (even if you don't call it)
1.448 + // Andy - all draw commands add to a gdi dirty region (which allocates)
1.449 + // Andy - combining client clipping regions with window clipping regions allocates
1.450 + scheduler = &fallbackScheduler;
1.451 + iScreen.FallbackMap()->Prepare();
1.452 + iScreen.RootWindow()->WalkWindowTree(fallbackScheduler,EWalkChildren);
1.453 + }
1.454 +
1.455 + CWsActiveScheduler::Static()->StartDraw();
1.456 + CWsMemoryManager::Static()->EnableReserve();
1.457 +
1.458 + if (&fallbackScheduler == scheduler)
1.459 + iAnimationRegion = iScreen.FallbackMap()->Region();
1.460 + else
1.461 + iAnimationRegion = &iInvalid;
1.462 +
1.463 + // Redraw debug regions more brightly than before:
1.464 + DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x80),&iInvalid);
1.465 + DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x80),&iTopLayer);
1.466 +
1.467 + RWsRegion accumulatedDrawing;
1.468 +
1.469 + // Pipe the drawing into the first render stage:
1.470 + CFbsBitGc * stageGc = iRenderStages->Begin();
1.471 +
1.472 + for (CWsWindow * win = scheduler->HeadWindow(); win; win = win->NextScheduled())
1.473 + {
1.474 + const TRegion * targetRegion = scheduler->Region(win);
1.475 + const TRect * screenRect = 0;
1.476 + if ((&fallbackScheduler == scheduler) && !targetRegion->IsContainedBy(iScreen.RootWindow()->Abs()))
1.477 + {
1.478 + screenRect = &iScreen.RootWindow()->Abs();
1.479 + }
1.480 + if (!screenRect)
1.481 + {
1.482 + // Purple regions are about to be drawn
1.483 + DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80),targetRegion);
1.484 + // Do the drawing
1.485 + stageGc->Reset();
1.486 + win->Render(stageGc, *targetRegion);
1.487 + accumulatedDrawing.Union(*targetRegion);
1.488 + // Green regions have been drawn
1.489 + DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80),targetRegion);
1.490 + }
1.491 + else
1.492 + {
1.493 + // Our region overlaps the edges of the screen, and we have no memory
1.494 + // to create a clipped one, so we will use a single-rect region for each rect
1.495 + // and call Draw multiple times.
1.496 + TRegionFix<1> rectRegion;
1.497 + for (const TRect * rect = targetRegion->RectangleList() + targetRegion->Count() - 1; rect >= targetRegion->RectangleList(); --rect)
1.498 + {
1.499 + rectRegion.Clear();
1.500 + TRect combined(*screenRect);
1.501 + combined.Intersection(*rect);
1.502 + rectRegion.AddRect(combined);
1.503 + // Purple regions are about to be drawn
1.504 + DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80),&rectRegion);
1.505 + // Do the drawing
1.506 + stageGc->Reset();
1.507 + win->Render(stageGc, rectRegion);
1.508 + accumulatedDrawing.AddRect(combined);
1.509 + // Green regions have been drawn
1.510 + DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80),&rectRegion);
1.511 + }
1.512 + }
1.513 + DEBUGOSB
1.514 + }
1.515 +
1.516 + DEBUGOSB
1.517 + iScreen.SpriteManager()->DrawFloatingSprites(stageGc,accumulatedDrawing); // we limit sprite drawing over only actually affected ares but necessary to all "planned" for redraw
1.518 + if (!accumulatedDrawing.CheckError() && iScreen.SpriteManager()->SpriteCount() == 0)
1.519 + {
1.520 + iAnimationRegion = &accumulatedDrawing;
1.521 + }
1.522 +
1.523 + // Tell the render stage we've finished:
1.524 + iRenderStages->End();
1.525 +
1.526 + // We nolonger need the regions
1.527 + for (CWsWindow * win = scheduler->HeadWindow(); win; win = win->NextScheduled())
1.528 + {
1.529 + win->ClearScheduledRegion();
1.530 + }
1.531 +
1.532 + CWsMemoryManager::Static()->DisableReserve();
1.533 +
1.534 + if (const CDebugBar* dbg = iScreen.DebugBar())
1.535 + {
1.536 + dbg->RedrawDebugBar();
1.537 + }
1.538 +
1.539 + // At this point, if the DEBUG_REGION is being used, there should usually be only green regions
1.540 + // displayed. If we see red or yellow, then something didn't get redrawn that should have been.
1.541 + // If we see purple then a window has disobeyed the const setting on the region to draw.
1.542 + // Red or yellow is valid - a window can decide it isn't ready to draw yet - purple is very bad.
1.543 + iScreen.Update();
1.544 + // At this point, if the DEBUG_REGION is being used, there should be no regions visible
1.545 + // of any colour. If we see green, then it means an area of the screen was drawn to which
1.546 + // wasn't invalid, or the screen update call failed. The former is more likely.
1.547 + // If we still see red or yellow it is a region that is not yet ready to draw.
1.548 +
1.549 + const TRect* rect = iAnimationRegion->RectangleList();
1.550 + TInt pixels = 0;
1.551 + for(TInt r = iAnimationRegion->Count(); r>0; r--)
1.552 + {
1.553 + pixels += (rect->Width()*rect->Height());
1.554 + rect++;
1.555 + }
1.556 + CWsActiveScheduler::Static()->StopDraw(pixels);
1.557 +
1.558 + TWindowServerEvent::NotifyScreenDrawingEvent(iAnimationRegion);
1.559 +
1.560 + iAnimationRegion = 0;
1.561 + accumulatedDrawing.Close();
1.562 + iInvalid.Clear();
1.563 + }
1.564 + else
1.565 + {
1.566 + CWsActiveScheduler::Static()->CancelPrepare();
1.567 + }
1.568 +
1.569 + iInvalid.Clear();
1.570 + iTopLayer.Clear();
1.571 + invalidCopy.Close();
1.572 + iAnimating = EFalse;
1.573 +
1.574 + if (futureAnimationRequired && iTimedDrawRect.Count() > 0 && !iScheduled)
1.575 + {
1.576 + // If this flag is set then it means there were already some animations scheduled when we ran,
1.577 + // but they themselves didn't run. We need to make sure we have _something_ scheduled.
1.578 + CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen, iTimedDrawRect[0].iTime);
1.579 + iScheduled = ETrue;
1.580 + }
1.581 +
1.582 + if(iObserver)
1.583 + {
1.584 + iObserver->ScreenUpdated(iScreen.ScreenNumber());
1.585 + iObserver=NULL; //once signalled we are never going to call it again
1.586 + }
1.587 + LOG_SCREEN_REDRAW_END
1.588 + }
1.589 +
1.590 +//
1.591 +void CScreenRedraw::DiscardAllSchedules()
1.592 + {
1.593 + ASSERT(!iAnimating);
1.594 +
1.595 + iTimedDrawRect.Reset();
1.596 + iInvalid.Clear();
1.597 + iInvalid.Tidy();
1.598 + }
1.599 +
1.600 +/**
1.601 +Indicates that a window has moved or changed ordinal position so that the visible regions
1.602 +of all windows needs to be recalculated
1.603 +*/
1.604 +void CScreenRedraw::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
1.605 + {
1.606 + iRegionUpdateScheduled = ETrue;
1.607 + ScheduleRedraw();
1.608 + if(aDefinitelyDirty)
1.609 + {
1.610 + iInvalid.Union(*aDefinitelyDirty);
1.611 + // Cyan regions for invalidations caused by this:
1.612 + DEBUG_REGION(TRgb(0x00, 0xFF, 0xFF),TRgb(0x00, 0xFF, 0xFF, 0x20),aDefinitelyDirty);
1.613 + }
1.614 + }
1.615 +
1.616 +/**
1.617 +Recalculates visible regions and schedules redraws or queues redraw events in response to
1.618 +any changes
1.619 +*/
1.620 +void CScreenRedraw::RegionUpdate()
1.621 + {
1.622 + if (iRegionUpdateScheduled)
1.623 + {
1.624 + iRegionUpdateScheduled = EFalse;
1.625 +
1.626 + TWalkWindowTreeUpdateRegions updater(iScreen);
1.627 + updater.Walk();
1.628 +
1.629 + WsPointer::ReLogCurrentWindow();
1.630 + CWsTop::TriggerRedraws(iScreen.RootWindow());
1.631 + }
1.632 + }
1.633 +
1.634 +void CScreenRedraw::SetObserver(MWsScreenRedrawObserver* aObserver)
1.635 + {
1.636 + iObserver = aObserver;
1.637 + }
1.638 +
1.639 +TBool CScreenRedraw::IsUpdatePending()
1.640 + {
1.641 + if(iScheduled || iAnimating)
1.642 + return ETrue;
1.643 + else
1.644 + return EFalse;
1.645 + }
1.646 +
1.647 +/**
1.648 + Overidding MWsObjectProvider
1.649 +*/
1.650 +TAny* CScreenRedraw::ResolveObjectInterface(TUint aTypeId)
1.651 + {
1.652 + TAny* interface = NULL;
1.653 +
1.654 + switch (aTypeId)
1.655 + {
1.656 + case KWsScreenRedraw:
1.657 + interface = static_cast<MWsScreenRedraw*>(this);
1.658 + break;
1.659 + }
1.660 +
1.661 + if (!interface)
1.662 + {
1.663 + interface = iRenderStages->ResolveObjectInterface(aTypeId);
1.664 + }
1.665 +
1.666 + return interface;
1.667 + }
1.668 +
1.669 +const TRegion * CScreenRedraw::AnimationRegion() const
1.670 + {
1.671 + if (iAnimating)
1.672 + return iAnimationRegion;
1.673 + else
1.674 + return 0;
1.675 + }
1.676 +
1.677 +void CScreenRedraw::UpdateDevice()
1.678 + {
1.679 + iScreen.Update();
1.680 + }
1.681 +
1.682 +void CScreenRedraw::BanThisRegionUpdate( TRegion& aForbiddenRegion )
1.683 + {
1.684 + iBannedRegion.Union( aForbiddenRegion );
1.685 + iBannedRegion.Tidy();
1.686 + }
1.687 +
1.688 +void CScreenRedraw::LiftRegionUpdateBan( TRegion& aFormerForbiddenRegion )
1.689 + {
1.690 + iBannedRegion.SubRegion( aFormerForbiddenRegion );
1.691 + iBannedRegion.Tidy();
1.692 + }
1.693 +
1.694 +void CScreenRedraw::AcceptFadeRequest( CWsWindow* aWin, TBool aFadeOn, TBool aFadeBehind, TBool aIncludeChildren )
1.695 + {
1.696 + if ( aFadeOn )
1.697 + {
1.698 + TWalkWindowTreeScheduleFadeNoRedraw walkerFadeNoRedraw;
1.699 + if ( aFadeBehind )
1.700 + {
1.701 + aWin->WalkWindowTree( walkerFadeNoRedraw, EWalkBehind );
1.702 + }
1.703 + else
1.704 + {
1.705 + if(aWin->WinType() != EWinTypeGroup)
1.706 + {
1.707 + ScheduleRegionUpdate( aWin->VisibleRegionIfValid() );
1.708 + }
1.709 +
1.710 + if ( aIncludeChildren )
1.711 + {
1.712 + aWin->WalkWindowTree( walkerFadeNoRedraw, EWalkChildren );
1.713 + }
1.714 + }
1.715 + }
1.716 + else
1.717 + { // fade off, just initiate redraw
1.718 + TWalkWindowTreeScheduleRedraws walkerRedraw( TWalkWindowTreeScheduleRedraws::ERedrawFilterOmitDSA ) ;
1.719 + if ( aFadeBehind )
1.720 + {
1.721 + aWin->WalkWindowTree( walkerRedraw, EWalkBehind );
1.722 + }
1.723 + else
1.724 + { // fade this win not behind
1.725 + if ( !aWin->IsDSAHost() )
1.726 + {
1.727 + AddRedrawRegion( aWin->VisibleRegion() );
1.728 + }
1.729 + if ( aIncludeChildren )
1.730 + {
1.731 + aWin->WalkWindowTree( walkerRedraw, EWalkChildren );
1.732 + }
1.733 + }
1.734 + }
1.735 + ScheduleRegionUpdate(NULL);
1.736 + }