Update contrib.
1 // Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
18 #include "ScreenRedraw.h"
25 #include "offscreenbitmap.h"
26 #include "wspluginmanager.h"
29 #include "walkwindowtree.h"
32 #include "Graphics/WsRenderStageFactory.h"
33 #include "Graphics/WsRenderStage.h"
36 GLREF_D CDebugLogBase *wsDebugLog;
38 #ifdef USE_DEBUG_REGIONS
39 # define DEBUG_REGION(col,fill,reg) DebugRegion(col,fill,reg)
40 # define DEBUG_RECT(col,fill,rect) DebugRect(col,fill,rect)
42 # define DEBUG_REGION(col,fill,reg)
43 # define DEBUG_RECT(col,fill,rect)
46 #if defined(__WINS__) && defined(_DEBUG)
47 # define DEBUGOSB { CWsOffScreenBitmap * ofb = iScreen.OffScreenBitmap(); if (ofb) ofb->Update(); }
53 # define LOG_SCREEN_REDRAW_START {if (wsDebugLog) {_LIT(KLogScreenRedrawStart, ">> CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawStart);}}
54 # define LOG_SCREEN_REDRAW_END {if (wsDebugLog) {_LIT(KLogScreenRedrawEnd, "<< CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawEnd);}}
56 # define LOG_SCREEN_REDRAW_START
57 # define LOG_SCREEN_REDRAW_END
60 CScreenRedraw::TTimedRect::TTimedRect(const TRect& aRect, const TTime& aTime):
61 iRect(aRect), iTime(aTime)
65 TInt CScreenRedraw::TTimedRect::Compare(const TTimedRect& aOne,const TTimedRect& aOther)
67 if(aOne.iTime < aOther.iTime)
69 else if(aOne.iTime > aOther.iTime)
75 CScreenRedraw * CScreenRedraw::NewL(CScreen& aScreen)
77 CScreenRedraw * self = new (ELeave) CScreenRedraw(aScreen);
78 CleanupStack::PushL(self);
80 CleanupStack::Pop(self);
84 CScreenRedraw::CScreenRedraw(CScreen& aScreen): iScreen(aScreen)
88 CScreenRedraw::~CScreenRedraw()
90 CWsRenderStage * stage = iRenderStages;
93 CWsRenderStage *next=stage->Next();
97 iTimedDrawRect.Close();
100 iBannedRegion.Close();
103 void CScreenRedraw::ConstructL()
105 CWsPluginManager * pluginMgr = CWsTop::WindowServer()->PluginManager();
107 // Setup the render stages for this screen:
108 _LIT(KDefaultRenderStages, "std");
109 _LIT(KDefaultFlickerFreeRenderStages, "flickerbuffer std");
110 _LIT(KRenderStages,"RENDERSTAGES");
112 const TBool customStages = WsIniFile->FindVar(iScreen.ScreenNumber(),KRenderStages,stagesString);
114 // If noone specifies stages for this screen, assume the standard implementation:
115 const TDesC * stages;
117 stages = &stagesString;
118 else if (iScreen.OffScreenBitmap())
119 stages = &KDefaultFlickerFreeRenderStages();
121 stages = &KDefaultRenderStages();
123 CWsRenderStage * lastStage = 0;
125 // Parse the string for implementation IDs:
129 TPtrC ptr = lex.NextToken();
130 if (ptr.Length() > 0)
133 MWsRenderStageFactory * factory = pluginMgr->FindNamedImplementation<MWsRenderStageFactory>(ptr);
136 CWsRenderStage * stage = 0;
137 TRAP(err, stage = factory->CreateStageL(static_cast<MWsScreen*>(&iScreen), this));
147 lastStage->SetNext(stage);
149 iRenderStages = stage;
164 _LIT(KAddedRenderStage,"Added render stage: ");
165 buf.Append(KAddedRenderStage);
167 wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,buf);
171 _LIT(KMissingRenderStage,"Failed to add render stage (%d): ");
172 buf.Append(KMissingRenderStage);
174 wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,buf,err);
185 const TTime& CScreenRedraw::Now() const
189 iNow.UniversalTime();
194 void CScreenRedraw::ScheduleRender(const TTimeIntervalMicroSeconds& aFromNow)
196 iRenderScheduled = ETrue;
197 TTime then(Now() + aFromNow);
198 if ((!iScheduled) || then < iNext)
201 CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
204 void CScreenRedraw::ScheduleRedraw()
209 // The other scheduler also removes future animations which this one encompasses.
210 // We choose not to do the region calculations needed to achieve that here.
211 MWsAnimationScheduler* animSched=CWsTop::WindowServer()->AnimationScheduler();
214 animSched->ScheduleRedraw(iScreen,iNext);
218 void CScreenRedraw::ScheduleAnimation(const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow,const TTimeIntervalMicroSeconds& /*aFreq*/,const TTimeIntervalMicroSeconds& /*aStop*/)
221 test.Intersection(iScreen.DrawableArea());
224 const TTime then(Now() + aFromNow);
225 TTimedRect tRect(aRect, then);
227 const TInt error = iTimedDrawRect.InsertInOrderAllowRepeats(tRect,TTimedRect::Compare);
228 if (KErrNone == error)
242 // remove further futures that are completely contained
243 TInt count = iTimedDrawRect.Count();
244 for(TInt i=0; i<count; i++)
246 const TTimedRect& future = iTimedDrawRect[i];
247 if(future.iTime.Int64() > then.Int64())
250 rect.BoundingRect(future.iRect);
251 if(rect == aRect) // future is completely contained within aRect
253 iTimedDrawRect.Remove(i);
259 CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
261 // Blue rectangles for scheduled animations
262 DEBUG_RECT(TRgb(0x00, 0x00, 0xFF),TRgb(0x00, 0x00, 0xFF, 0x20),&aRect);
267 // This adds a region to the stored invalid region.
268 // The invalid region is the area of the screen that needs to be redrawn in addition to any animations.
269 // The draw region is the area of the screen on which only the top window needs to be redrawn.
270 // If the top window has transparency, this can only be true when it has been made newly visible.
271 // The value of aSchedule could be determined automatically from iAnimating, but passing it this way
272 // allows us to have the assert, which is a very valuable assert.
273 void CScreenRedraw::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
275 WS_ASSERT_DEBUG(!iAnimating || !aSchedule, EWsPanicScheduledRedraw);
277 if(aRegion.CheckError())
279 iInvalid.ForceError();
284 else if(aRegion.Count()) // often called despite window not being visible
286 if (aDepth == ERedrawAll)
288 // red lines for an invalid region which is ready to be drawn
289 DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x20),&aRegion);
291 iInvalid.Union(aRegion);
298 // yellow lines for a valid region which we will draw on top of
299 DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x20),&aRegion);
301 iTopLayer.Union(aRegion);
309 // This causes any asynchronously scheduled redraw to happen immediately
310 // It should be avoided where possible for performance reasons, but is
311 // needed whenever the redraw store is discarded for a window which still
312 // has a redraw region pending.
313 void CScreenRedraw::DoRedrawNow()
316 CWsTop::WindowServer()->AnimationScheduler()->DoRedrawNow(iScreen);
319 #ifdef USE_DEBUG_REGIONS
320 void CScreenRedraw::DebugRect(TRgb aColor, TRgb aFill, const TRect* aRect)
324 CFbsBitGc * gc = iScreen.GetBitGc();
325 gc->SetPenColor(aColor);
326 gc->SetPenStyle(CGraphicsContext::ESolidPen);
327 gc->SetPenSize(TSize(2,2));
328 gc->SetBrushColor(aFill);
329 gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
330 TRect smaller = *aRect;
333 gc->DrawRect(smaller);
338 void CScreenRedraw::DebugRegion(TRgb aColor, TRgb aFill, const TRegion * aRegion)
342 CFbsBitGc * gc = iScreen.GetBitGc();
343 gc->SetPenColor(aColor);
344 gc->SetPenStyle(CGraphicsContext::ESolidPen);
345 gc->SetPenSize(TSize(2,2));
346 gc->SetBrushColor(aFill);
347 gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
348 for (const TRect *rect = aRegion->RectangleList(); rect - aRegion->RectangleList() < aRegion->Count(); ++rect)
350 TRect smaller = *rect;
353 gc->DrawRect(smaller);
360 void CScreenRedraw::OnAnimation()
362 LOG_SCREEN_REDRAW_START
367 TBool futureAnimationRequired = EFalse;
369 CWsActiveScheduler::Static()->PrepareDraw();
371 // Calculate any updates required by region changes:
374 // Add the timed rectangles to the invalid region:
375 iNow.UniversalTime();
376 TInt count(iTimedDrawRect.Count());
379 if(iTimedDrawRect[0].iTime.Int64() <= iNow.Int64())
381 iInvalid.AddRect(iTimedDrawRect[0].iRect);
382 iTimedDrawRect.Remove(0);
387 futureAnimationRequired = ETrue;
392 // Animating rectangles could cause iInvalid to overlap iTopLayer,
393 // in which case iTopLayer won't work.
394 iTopLayer.SubRegion(iInvalid);
395 iTopLayer.Intersect(iScreen.RootWindow()->WindowArea());
397 // Anything in the top layer is implcitly invalid
398 iInvalid.Union(iTopLayer);
399 iInvalid.Intersect(iScreen.RootWindow()->WindowArea());
402 if (const CDebugBar* dbg = iScreen.DebugBar())
404 iTopLayer.SubRect(dbg->Rect());
405 iInvalid.SubRect(dbg->Rect());
411 if(iInvalid.CheckError()) //Will: agree with Andy, want bounding rects instead!
415 iInvalid.Copy(iScreen.RootWindow()->WindowArea()); // assumed cannot fail, all regions can contain at least 1 rect..
418 iInvalid.SubRegion( iBannedRegion );
420 iTopLayer.SubRegion( iBannedRegion );
423 STACK_REGION invalidCopy;
424 invalidCopy.Copy(iInvalid);
425 TWalkWindowTreeScheduleRegions regionScheduler(&invalidCopy, iTopLayer);
426 TWalkWindowTreeScheduleFallback fallbackScheduler(iScreen.FallbackMap());
427 TWalkWindowTreeSchedule * scheduler = ®ionScheduler;
429 // At this point, if the DEBUG_REGION is being used:
430 // Red represents invalid regions that need to be redrawn completely.
431 // Yellow represents regions that only need the top window to be drawn.
432 // Blue represents regions which are being animated server side.
433 if (iRenderScheduled || !iInvalid.IsEmpty())
435 iRenderScheduled = EFalse;
436 // invalidCopy.ForceError(); //### DEBUG
438 iScreen.RootWindow()->WalkWindowTree(regionScheduler,EWalkChildren);
439 if (!regionScheduler.ScheduledRegionsOk())
441 // our region calculations for what to draw failed at some point.
442 // From this point on we MUST NOT rely on allocating memory
443 // Andy - localRedrawRegion allocates
444 // Andy - setPenSize allocates (even if you don't call it)
445 // Andy - all draw commands add to a gdi dirty region (which allocates)
446 // Andy - combining client clipping regions with window clipping regions allocates
447 scheduler = &fallbackScheduler;
448 iScreen.FallbackMap()->Prepare();
449 iScreen.RootWindow()->WalkWindowTree(fallbackScheduler,EWalkChildren);
452 CWsActiveScheduler::Static()->StartDraw();
453 CWsMemoryManager::Static()->EnableReserve();
455 if (&fallbackScheduler == scheduler)
456 iAnimationRegion = iScreen.FallbackMap()->Region();
458 iAnimationRegion = &iInvalid;
460 // Redraw debug regions more brightly than before:
461 DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x80),&iInvalid);
462 DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x80),&iTopLayer);
464 RWsRegion accumulatedDrawing;
466 // Pipe the drawing into the first render stage:
467 CFbsBitGc * stageGc = iRenderStages->Begin();
469 for (CWsWindow * win = scheduler->HeadWindow(); win; win = win->NextScheduled())
471 const TRegion * targetRegion = scheduler->Region(win);
472 const TRect * screenRect = 0;
473 if ((&fallbackScheduler == scheduler) && !targetRegion->IsContainedBy(iScreen.RootWindow()->Abs()))
475 screenRect = &iScreen.RootWindow()->Abs();
479 // Purple regions are about to be drawn
480 DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80),targetRegion);
483 win->Render(stageGc, *targetRegion);
484 accumulatedDrawing.Union(*targetRegion);
485 // Green regions have been drawn
486 DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80),targetRegion);
490 // Our region overlaps the edges of the screen, and we have no memory
491 // to create a clipped one, so we will use a single-rect region for each rect
492 // and call Draw multiple times.
493 TRegionFix<1> rectRegion;
494 for (const TRect * rect = targetRegion->RectangleList() + targetRegion->Count() - 1; rect >= targetRegion->RectangleList(); --rect)
497 TRect combined(*screenRect);
498 combined.Intersection(*rect);
499 rectRegion.AddRect(combined);
500 // Purple regions are about to be drawn
501 DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80),&rectRegion);
504 win->Render(stageGc, rectRegion);
505 accumulatedDrawing.AddRect(combined);
506 // Green regions have been drawn
507 DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80),&rectRegion);
514 iScreen.SpriteManager()->DrawFloatingSprites(stageGc,accumulatedDrawing); // we limit sprite drawing over only actually affected ares but necessary to all "planned" for redraw
515 if (!accumulatedDrawing.CheckError() && iScreen.SpriteManager()->SpriteCount() == 0)
517 iAnimationRegion = &accumulatedDrawing;
520 // Tell the render stage we've finished:
521 iRenderStages->End();
523 // We nolonger need the regions
524 for (CWsWindow * win = scheduler->HeadWindow(); win; win = win->NextScheduled())
526 win->ClearScheduledRegion();
529 CWsMemoryManager::Static()->DisableReserve();
531 if (const CDebugBar* dbg = iScreen.DebugBar())
533 dbg->RedrawDebugBar();
536 // At this point, if the DEBUG_REGION is being used, there should usually be only green regions
537 // displayed. If we see red or yellow, then something didn't get redrawn that should have been.
538 // If we see purple then a window has disobeyed the const setting on the region to draw.
539 // Red or yellow is valid - a window can decide it isn't ready to draw yet - purple is very bad.
541 // At this point, if the DEBUG_REGION is being used, there should be no regions visible
542 // of any colour. If we see green, then it means an area of the screen was drawn to which
543 // wasn't invalid, or the screen update call failed. The former is more likely.
544 // If we still see red or yellow it is a region that is not yet ready to draw.
546 const TRect* rect = iAnimationRegion->RectangleList();
548 for(TInt r = iAnimationRegion->Count(); r>0; r--)
550 pixels += (rect->Width()*rect->Height());
553 CWsActiveScheduler::Static()->StopDraw(pixels);
555 TWindowServerEvent::NotifyScreenDrawingEvent(iAnimationRegion);
557 iAnimationRegion = 0;
558 accumulatedDrawing.Close();
563 CWsActiveScheduler::Static()->CancelPrepare();
571 if (futureAnimationRequired && iTimedDrawRect.Count() > 0 && !iScheduled)
573 // If this flag is set then it means there were already some animations scheduled when we ran,
574 // but they themselves didn't run. We need to make sure we have _something_ scheduled.
575 CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen, iTimedDrawRect[0].iTime);
581 iObserver->ScreenUpdated(iScreen.ScreenNumber());
582 iObserver=NULL; //once signalled we are never going to call it again
584 LOG_SCREEN_REDRAW_END
588 void CScreenRedraw::DiscardAllSchedules()
592 iTimedDrawRect.Reset();
598 Indicates that a window has moved or changed ordinal position so that the visible regions
599 of all windows needs to be recalculated
601 void CScreenRedraw::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
603 iRegionUpdateScheduled = ETrue;
607 iInvalid.Union(*aDefinitelyDirty);
608 // Cyan regions for invalidations caused by this:
609 DEBUG_REGION(TRgb(0x00, 0xFF, 0xFF),TRgb(0x00, 0xFF, 0xFF, 0x20),aDefinitelyDirty);
614 Recalculates visible regions and schedules redraws or queues redraw events in response to
617 void CScreenRedraw::RegionUpdate()
619 if (iRegionUpdateScheduled)
621 iRegionUpdateScheduled = EFalse;
623 TWalkWindowTreeUpdateRegions updater(iScreen);
626 WsPointer::ReLogCurrentWindow();
627 CWsTop::TriggerRedraws(iScreen.RootWindow());
631 void CScreenRedraw::SetObserver(MWsScreenRedrawObserver* aObserver)
633 iObserver = aObserver;
636 TBool CScreenRedraw::IsUpdatePending()
638 if(iScheduled || iAnimating)
645 Overidding MWsObjectProvider
647 TAny* CScreenRedraw::ResolveObjectInterface(TUint aTypeId)
649 TAny* interface = NULL;
653 case KWsScreenRedraw:
654 interface = static_cast<MWsScreenRedraw*>(this);
660 interface = iRenderStages->ResolveObjectInterface(aTypeId);
666 const TRegion * CScreenRedraw::AnimationRegion() const
669 return iAnimationRegion;
674 void CScreenRedraw::UpdateDevice()
679 void CScreenRedraw::BanThisRegionUpdate( TRegion& aForbiddenRegion )
681 iBannedRegion.Union( aForbiddenRegion );
682 iBannedRegion.Tidy();
685 void CScreenRedraw::LiftRegionUpdateBan( TRegion& aFormerForbiddenRegion )
687 iBannedRegion.SubRegion( aFormerForbiddenRegion );
688 iBannedRegion.Tidy();
691 void CScreenRedraw::AcceptFadeRequest( CWsWindow* aWin, TBool aFadeOn, TBool aFadeBehind, TBool aIncludeChildren )
695 TWalkWindowTreeScheduleFadeNoRedraw walkerFadeNoRedraw;
698 aWin->WalkWindowTree( walkerFadeNoRedraw, EWalkBehind );
702 if(aWin->WinType() != EWinTypeGroup)
704 ScheduleRegionUpdate( aWin->VisibleRegionIfValid() );
707 if ( aIncludeChildren )
709 aWin->WalkWindowTree( walkerFadeNoRedraw, EWalkChildren );
714 { // fade off, just initiate redraw
715 TWalkWindowTreeScheduleRedraws walkerRedraw( TWalkWindowTreeScheduleRedraws::ERedrawFilterOmitDSA ) ;
718 aWin->WalkWindowTree( walkerRedraw, EWalkBehind );
721 { // fade this win not behind
722 if ( !aWin->IsDSAHost() )
724 AddRedrawRegion( aWin->VisibleRegion() );
726 if ( aIncludeChildren )
728 aWin->WalkWindowTree( walkerRedraw, EWalkChildren );
732 ScheduleRegionUpdate(NULL);