First public contribution.
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.
16 #include "ScreenRedraw.h"
25 #include "walkwindowtree.h"
28 #include "renderstagemanager.h"
29 #include "graphics/WsRenderStageFactory.h"
30 #include "graphics/WsRenderStage.h"
31 #include "graphics/wsgraphicscontext.h"
34 #ifdef USE_DEBUG_FRAME_CAPTURE
35 #include <graphics/wsscreendevice.h>
36 #include "../debuglog/debuglog.h"
39 GLREF_D CDebugLogBase *wsDebugLog;
41 #ifdef USE_DEBUG_REGIONS
42 # define DEBUG_REGION(col,fill,reg) DebugRegion(col,fill,reg)
43 # define DEBUG_RECT(col,fill,rect) DebugRect(col,fill,rect)
45 # define DEBUG_REGION(col,fill,reg)
46 # define DEBUG_RECT(col,fill,rect)
50 # define LOG_SCREEN_REDRAW_START {if (wsDebugLog) {_LIT(KLogScreenRedrawStart, ">> CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawStart);}}
51 # define LOG_SCREEN_REDRAW_END {if (wsDebugLog) {_LIT(KLogScreenRedrawEnd, "<< CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawEnd);}}
53 # define LOG_SCREEN_REDRAW_START
54 # define LOG_SCREEN_REDRAW_END
58 #ifdef USE_DEBUG_FRAME_CAPTURE
60 _LIT(KDefaultFrameCaptureLocation, "c:\\");
62 _LIT(KDefaultFrameCaptureLocation, "e:\\");
64 #endif //USE_DEBUG_FRAME_CAPTURE
66 CScreenRedraw::TTimedRect::TTimedRect(TAnimType aType, const TRect& aRect, const TTime& aTime, CWsWindow* aWindow)
67 : iType(aType), iRect(aRect), iTime(aTime), iWindow(aWindow)
71 TInt CScreenRedraw::TTimedRect::Compare(const TTimedRect& aOne,const TTimedRect& aOther)
73 if(aOne.iTime < aOther.iTime)
75 else if(aOne.iTime > aOther.iTime)
81 CScreenRedraw * CScreenRedraw::NewL(CScreen& aScreen)
83 CScreenRedraw * self = new (ELeave) CScreenRedraw(aScreen);
84 CleanupStack::PushL(self);
86 CleanupStack::Pop(self);
90 CScreenRedraw::CScreenRedraw(CScreen& aScreen): iScreen(aScreen)
94 CScreenRedraw::~CScreenRedraw()
96 CRenderStageManager::Release(iRenderStages);
98 iTimedDrawRect.Close();
101 iQuickFadeList.Reset();
102 #ifdef USE_DEBUG_FRAME_CAPTURE
104 delete iDebugBitmapDevice;
105 delete iDebugBitmapGc;
106 #endif //USE_DEBUG_FRAME_CAPTURE
109 void CScreenRedraw::ConstructL()
114 iRenderStages = CRenderStageManager::ConnectL(
115 iScreen.ScreenNumber(),
116 static_cast<MWsScreen*>(&iScreen),
120 LEAVE_LOG_UNINSTALL_C;
122 WS_ASSERT_ALWAYS(iRenderStages, EWsPanicNoRenderStagePipeline);
123 iRenderStageTextCursor = static_cast<MWsTextCursor*>(iRenderStages->ResolveObjectInterface(KMWsTextCursor));
124 WS_ASSERT_ALWAYS(iRenderStageTextCursor, EWsPanicTextCursorInterfaceMissing);
126 #ifdef USE_DEBUG_FRAME_CAPTURE
127 _LIT(KWSERVIniFileVarFrameCapture,"FRAMECAPTURE");
128 iFrameCapture = WsIniFile->FindVar(KWSERVIniFileVarFrameCapture);
130 // Location to save captured images
131 if (!WsIniFile->FindVar(KWSERVIniFileVarFrameCapture, iFrameCaptureLocation) || iFrameCaptureLocation.Length() == 0)
133 iFrameCaptureLocation.Set(KDefaultFrameCaptureLocation);
135 #endif //USE_DEBUG_FRAME_CAPTURE
138 MWsTextCursor* CScreenRedraw::RenderStageTextCursor() const
140 return iRenderStageTextCursor;
143 const TTime& CScreenRedraw::Now() const
147 iNow.UniversalTime();
152 void CScreenRedraw::ScheduleRender(const TTimeIntervalMicroSeconds& aFromNow)
154 iRenderScheduled = ETrue;
155 TTime then(Now() + aFromNow);
156 if ((!iScheduled) || then < iNext)
159 CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
162 void CScreenRedraw::ScheduleRedraw()
167 // The other scheduler also removes future animations which this one encompasses.
168 // We choose not to do the region calculations needed to achieve that here.
169 CWindowServer* server = CWsTop::WindowServer();
171 MWsAnimationScheduler* sched = server->AnimationScheduler();
173 sched->ScheduleRedraw(iScreen,iNext);
175 RDebug::Printf("CWsTop::WindowServer()->RedrawScheduler() is NULL!!!");
179 @param aRect in screen coordinates
181 void CScreenRedraw::ScheduleAnimation(TAnimType aType, const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow,const TTimeIntervalMicroSeconds& /*aFreq*/,const TTimeIntervalMicroSeconds& /*aStop*/, CWsWindow* aWindow)
183 TRect scheduledRect(aRect);
185 // In changetracking mode, animation is either scheduled via a window dirty region (aWindow non-NULL)
186 // or via a sprite manager dirty region (aWindow NULL) if scheduling a floating sprite
187 if (iScreen.ChangeTracking())
191 // scheduling a window dirty rect for a window/anim/sprite
192 test.Intersection(aWindow->Abs());
193 scheduledRect.Move(-aWindow->Origin()); // convert to window coordinates
195 // else, // scheduling a sprite manager dirty rect for a floating sprite
199 // scheduling a screen dirty rect
200 test.Intersection(iScreen.DrawableArea());
201 aWindow = NULL; // ensure all future timed draw screen rects are checked below
205 const TTime then(Now() + aFromNow);
206 TTimedRect tRect(aType, scheduledRect, then, aWindow);
208 const TInt error = iTimedDrawRect.InsertInOrderAllowRepeats(tRect,TTimedRect::Compare);
209 if (KErrNone == error)
223 // remove further futures that are completely contained
224 TInt count = iTimedDrawRect.Count();
225 for(TInt i=0; i<count; i++)
227 const TTimedRect& future = iTimedDrawRect[i];
228 if (future.iWindow != aWindow) // only check futures for the window/floating sprite/screen we're scheduling
230 if(future.iTime.Int64() > then.Int64())
232 TRect rect(scheduledRect);
233 rect.BoundingRect(future.iRect);
234 if(rect == scheduledRect) // future is completely contained within scheduledRect
236 iTimedDrawRect.Remove(i);
242 CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
244 // Blue rectangles for scheduled animations
245 DEBUG_RECT(TRgb(0x00, 0x00, 0xFF),TRgb(0x00, 0x00, 0xFF, 0x20),&aRect);
250 TBool CScreenRedraw::IsScheduled(TAnimType aType, const TRect& aRect, CWsWindow* aWindow) const
255 rect.Move(-aWindow->Origin()); //convert to window coordinates
258 const TInt count(iTimedDrawRect.Count());
259 for(TInt i=0; i<count; i++)
261 if ((iTimedDrawRect[i].iType == aType) &&
262 (iTimedDrawRect[i].iRect == rect) &&
263 (iTimedDrawRect[i].iWindow == aWindow))
271 // This adds a region to the stored invalid region.
272 // The invalid region is the area of the screen that needs to be redrawn in addition to any animations.
273 // The draw region is the area of the screen on which only the top window needs to be redrawn.
274 // If the top window has transparency, this can only be true when it has been made newly visible.
275 // The value of aSchedule could be determined automatically from iAnimating, but passing it this way
276 // allows us to have the assert, which is a very valuable assert.
277 void CScreenRedraw::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
279 WS_ASSERT_DEBUG(!iAnimating || !aSchedule, EWsPanicScheduledRedraw);
281 if(aRegion.CheckError())
283 iInvalid.ForceError();
288 else if(aRegion.Count()) // often called despite window not being visible
290 if (aDepth == ERedrawAll)
292 // red lines for an invalid region which is ready to be drawn
293 DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x20),&aRegion);
294 iInvalid.Union(aRegion);
301 // yellow lines for a valid region which we will draw on top of
302 DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x20),&aRegion);
304 iTopElement.Union(aRegion);
312 // This causes any asynchronously scheduled redraw to happen immediately
313 // It should be avoided where possible for performance reasons, but is
314 // needed whenever the redraw store is discarded for a window which still
315 // has a redraw region pending.
316 // @note This method always attempts to DoRedrawNow(), even if currently
317 // animating. In this context, animation is a frame update. With NGA
318 // that update completes at some point in the future. If you get a request
319 // to DoRedrawNow() the expectation is that all updates scheduled on or
320 // before that point in time will have completed by the time the call
321 // returns. An update may have been scheduled during a current (asynchronous)
322 // Animate call...the animation scheduler has this knowledge, so let it
323 // decide what to do.
325 void CScreenRedraw::DoRedrawNow()
327 CWsTop::WindowServer()->AnimationScheduler()->DoRedrawNow(iScreen);
330 #ifdef USE_DEBUG_REGIONS
331 void CScreenRedraw::DebugRect(TRgb aColor, TRgb aFill, const TRect* aRect)
335 CFbsBitGc * gc = iScreen.GetBitGc();
336 gc->SetPenColor(aColor);
337 gc->SetPenStyle(CGraphicsContext::ESolidPen);
338 gc->SetPenSize(TSize(2,2));
339 gc->SetBrushColor(aFill);
340 gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
341 TRect smaller = *aRect;
344 gc->DrawRect(smaller);
349 void CScreenRedraw::DebugRegion(TRgb aColor, TRgb aFill, const TRegion * aRegion)
353 CFbsBitGc * gc = iScreen.GetBitGc();
354 gc->SetPenColor(aColor);
355 gc->SetPenStyle(CGraphicsContext::ESolidPen);
356 gc->SetPenSize(TSize(2,2));
357 gc->SetBrushColor(aFill);
358 gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
359 for (const TRect *rect = aRegion->RectangleList(); rect - aRegion->RectangleList() < aRegion->Count(); ++rect)
361 TRect smaller = *rect;
364 gc->DrawRect(smaller);
371 void CScreenRedraw::OnAnimation(TRequestStatus* aFinished)
373 LOG_SCREEN_REDRAW_START
374 WS_ASSERT_ALWAYS(!iAnimating,EWsPanicAnimationAlreadyAnimating);
375 WS_ASSERT_ALWAYS(iScheduled,EWsPanicAnimationNotScheduled);
379 const TBool changeTracking = iScreen.ChangeTracking();
381 const TRegionFix<1> fallbackRegion(iScreen.RootWindow()->Abs());
383 CWsActiveScheduler::Static()->PrepareDraw();
385 // Calculate any updates required by region changes:
388 // Use the timed rectangles to mark screen, window or floating sprite as dirty
389 RWsRegionBuf<10> floatingSpriteDirtyRegion;
390 const TBool futureAnimationRequired = ScheduleTimedRects(floatingSpriteDirtyRegion);
391 if (floatingSpriteDirtyRegion.CheckError())
393 floatingSpriteDirtyRegion.Reset();
394 floatingSpriteDirtyRegion.Copy(fallbackRegion);
397 TWalkWindowTreeSchedule* scheduler = NULL;
398 TWalkWindowListSchedule windowScheduler(iScheduledWindowList, iInvalid); //ChangeTracking
401 // Animating rectangles could cause iInvalid to overlap iTopElement, in which case iTopElement won't work.
402 iTopElement.SubRegion(iInvalid);
403 iTopElement.Intersect(iScreen.RootWindow()->WindowArea());
405 // Anything in the top element is implicitly invalid
406 iInvalid.Union(iTopElement);
407 iInvalid.Intersect(iScreen.RootWindow()->WindowArea());
410 if(iInvalid.CheckError())
414 iInvalid.Copy(fallbackRegion);
419 // In ChangeTracking mode, iInvalid is only used to tell the render stage
420 // which part of the screen needs updating.
422 iInvalid.Copy(floatingSpriteDirtyRegion);
423 scheduler = &windowScheduler;
424 windowScheduler.WalkWindowList();
425 CWsTop::TriggerRedraws(iScreen.RootWindow()); //In case WalkWindowList did queue a request for the client to provide draw commands
427 if(iInvalid.CheckError())
430 iInvalid.Copy(fallbackRegion);
434 // At this point, if the DEBUG_REGION is being used:
435 // Red represents invalid regions that need to be redrawn completely.
436 // Yellow represents regions that only need the top window to be drawn.
437 // Blue represents regions which are being animated server side.
438 if (iRenderScheduled || !iInvalid.IsEmpty() || iQuickFadeList.Count() > 0)
440 iRenderScheduled = EFalse;
442 TWalkWindowTreeScheduleRegions regionScheduler(iInvalid, iTopElement); //!ChangeTeacking
443 TWalkWindowTreeScheduleFallback fallbackScheduler(iScreen.FallbackMap());//!ChangeTeacking
445 RWsRegionBuf<20> animationRegion;
449 animationRegion.Copy(iInvalid);
450 AddQuickFadeableRegions(animationRegion);
451 if (animationRegion.CheckError())
453 animationRegion.Reset();
454 animationRegion.Copy(fallbackRegion);
456 iAnimationRegion = &animationRegion; //iAnimationRegion must be reset to NULL before the call stack unwinds.
458 scheduler = ®ionScheduler;
459 iScreen.RootWindow()->WalkWindowTree(regionScheduler,EWalkChildren);
460 if (!regionScheduler.ScheduledRegionsOk())
462 // our region calculations for what to draw failed at some point.
463 // From this point on we MUST NOT rely on allocating memory
464 scheduler = &fallbackScheduler;
465 iScreen.FallbackMap()->Prepare();
466 iScreen.RootWindow()->WalkWindowTree(fallbackScheduler,EWalkChildren);
467 iAnimationRegion = iScreen.FallbackMap()->Region();
472 iAnimationRegion = &iInvalid;
475 CWsActiveScheduler::Static()->StartDraw();
476 CWsMemoryManager::Static()->EnableReserve();
478 // Redraw debug regions more brightly than before:
479 DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x80),iAnimationRegion);
480 DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x80),&iTopElement);
482 // Pipe the drawing into the first render stage:
483 iRenderStages->Begin(iAnimationRegion);
485 RWsRegionBuf<10> accumulatedDrawing;
487 MWsGraphicsContext * stageGc = static_cast<MWsGraphicsContext*>(iRenderStages->ResolveObjectInterface(KMWsGraphicsContext));
488 for (CWsWindow * win = scheduler->HeadWindow(); (win!=NULL); win = win->NextScheduled())
491 accumulatedDrawing.Union(scheduler->WindowRegion(*win));
493 Render(*win, *stageGc, *scheduler);
496 if(!changeTracking && !accumulatedDrawing.CheckError())
498 //Fade any region of the screen scheduled for fading which has not been redrawn,
499 DoQuickFade(stageGc, accumulatedDrawing);
501 accumulatedDrawing.Reset();
503 //We limit floating sprite drawing to regions already touched
504 iScreen.SpriteManager()->DrawFloatingSprites(stageGc, iScreen.ChangeTracking() ? floatingSpriteDirtyRegion : *iAnimationRegion);
506 iRenderStages->End(aFinished);
507 stageGc = NULL; //we're not allowed to draw outside Begin()/End()
509 #if defined(__WINS__) && defined(_DEBUG)
510 MWsDebugBar * debugBar = static_cast<MWsDebugBar*>(iRenderStages->ResolveObjectInterface(KMWsDebugBar));
511 if (debugBar) //optional for the licensees
513 if (CDebugBar* dbg = iScreen.DebugBar())
515 RArray<TPtrC> debugText;
516 dbg->DebugBarInfo(debugText);
517 debugBar->DrawDebugBar(debugText.Array());
523 iScheduledWindowList = NULL;
524 CWsMemoryManager::Static()->DisableReserve();
526 #ifdef USE_DEBUG_FRAME_CAPTURE
529 CaptureFrame(iAnimationRegion);
533 // DEBUG_REGION does not work with render stages.
534 // These comments refer to what used to happen.
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.
543 // At this point, if the DEBUG_REGION is being used, there should be no regions visible
544 // of any colour. If we see green, then it means an area of the screen was drawn to which
545 // wasn't invalid, or the screen update call failed. The former is more likely.
546 // If we still see red or yellow it is a region that is not yet ready to draw.
548 const TRect* rect = iAnimationRegion->RectangleList();
550 for(TInt r = iAnimationRegion->Count(); r>0; r--)
552 pixels += (rect->Width()*rect->Height());
555 CWsActiveScheduler::Static()->StopDraw(pixels);
557 TWindowServerEvent::NotifyScreenDrawingEvent(iAnimationRegion);
559 iAnimationRegion = NULL; //iAnimationRegion must be reset to NULL before the call stack unwinds.
561 animationRegion.Reset();
565 // There was nothing to compose. Signal that composition is complete.
568 *aFinished = KRequestPending;
569 User::RequestComplete(aFinished, KErrNone);
572 CWsActiveScheduler::Static()->CancelPrepare();
575 floatingSpriteDirtyRegion.Reset();
580 if (futureAnimationRequired && iTimedDrawRect.Count() > 0 && !iScheduled)
582 // If this flag is set then it means there were already some animations scheduled when we ran,
583 // but they themselves didn't run. We need to make sure we have _something_ scheduled.
584 CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen, iTimedDrawRect[0].iTime);
590 iObserver->ScreenUpdated(iScreen.ScreenNumber());
591 iObserver=NULL; //once signalled we are never going to call it again
593 LOG_SCREEN_REDRAW_END
596 void CScreenRedraw::AddQuickFadeableRegions(TRegion& aRegion)
598 if (iQuickFadeList.Count() > 0)
600 for (TInt idx = iQuickFadeList.Count() - 1; idx >= 0; idx --)
602 CWsWindow* win = iQuickFadeList[ idx ];
604 if ( !win->IsDSAHost() )
606 aRegion.Union(win->QuickFadeRegion() );
613 void CScreenRedraw::DoQuickFade(MWsGraphicsContext* aGc, TRegion& aAccumulatedDrawing)
615 if ( iQuickFadeList.Count() > 0 )
617 for ( TInt idx = iQuickFadeList.Count() - 1; idx >= 0 ; idx -- )
620 CWsWindow* win = iQuickFadeList[ idx ];
622 if ( !win->IsDSAHost() )
624 STACK_REGION winFadeRgn;
625 winFadeRgn.Copy( win->QuickFadeRegion() );
626 winFadeRgn.SubRegion( aAccumulatedDrawing );
628 if ( !winFadeRgn.IsEmpty() && !winFadeRgn.CheckError() )
629 { // now fade any region that has not been redrawn (via win->Render())
630 win->Fade( aGc, winFadeRgn );
631 aAccumulatedDrawing.Union( winFadeRgn );
637 iQuickFadeList.Reset();
642 This function will iterate the timed rectangles and find any animation or sprite
643 that are due to be scheduled for render now. Then schedule them and remove them
645 @return ETrue if iTimedDrawRect still contains future animations
647 TBool CScreenRedraw::ScheduleTimedRects(TRegion& aScheduledFloatingSpriteRegion)
649 iNow.UniversalTime();
650 TInt count(iTimedDrawRect.Count());
651 TBool futureAnimationRequired = EFalse;
654 if(iTimedDrawRect[0].iTime.Int64() <= iNow.Int64())
656 if (iScreen.ChangeTracking())
658 switch(iTimedDrawRect[0].iType)
662 CWsWindow* win = iTimedDrawRect[0].iWindow;
663 WS_ASSERT_DEBUG(win,EWsPanicWindowNull);
664 if(!win->IsTrackingVisibleRegion() || (win->IsTrackingVisibleRegion() && !win->VisibleRegion().IsEmpty()))
666 TRegionFix<1> region;
667 region.AddRect(iTimedDrawRect[0].iRect);
668 win->AddDirtyWindowRegion(region);
678 CWsWindow* win = iTimedDrawRect[0].iWindow;
679 WS_ASSERT_DEBUG(win,EWsPanicWindowNull);
680 if(!win->IsTrackingVisibleRegion() || (win->IsTrackingVisibleRegion() && !win->VisibleRegion().IsEmpty()))
682 TRegionFix<1> region;
683 region.AddRect(iTimedDrawRect[0].iRect);
684 win->AddDirtySpriteRegion(region);
689 case EFloatingSprite:
690 case EFloatingSpriteAnim:
692 aScheduledFloatingSpriteRegion.AddRect(iTimedDrawRect[0].iRect);
697 TRect screenRect(iTimedDrawRect[0].iRect);
698 if(iTimedDrawRect[0].iWindow)
700 screenRect.Move(iTimedDrawRect[0].iWindow->Origin()); // convert to screen coordinates
705 // schedule a screen dirty rect
706 iInvalid.AddRect(iTimedDrawRect[0].iRect);
708 iTimedDrawRect.Remove(0);
713 futureAnimationRequired = ETrue;
717 return futureAnimationRequired;
720 void CScreenRedraw::Render(CWsWindow& aWin, MWsGraphicsContext& aGc, const TWalkWindowTreeSchedule& aScheduler)
722 const TRegion* windowRegion = &(aScheduler.WindowRegion(aWin));
723 const TRegion* spriteRegion = &(aScheduler.SpriteRegion(aWin));
725 if(windowRegion->IsEmpty() && spriteRegion->IsEmpty())
726 return; //Don't call CWsWindow::Render if there is nothing that can be rendered
728 //Make sure we don't try to draw outside screen
729 STACK_REGION clippedWindowRegion;
730 if(!windowRegion->IsContainedBy(iScreen.RootWindow()->Abs()))
732 TRegionFix<1> screen(iScreen.RootWindow()->Abs());
733 clippedWindowRegion.Intersection(*windowRegion, screen);
734 windowRegion = &clippedWindowRegion;
736 STACK_REGION clippedSpriteRegion;
737 if(!spriteRegion->IsContainedBy(iScreen.RootWindow()->Abs()))
739 TRegionFix<1> screen(iScreen.RootWindow()->Abs());
740 clippedSpriteRegion.Intersection(*spriteRegion, screen);
741 spriteRegion = &clippedSpriteRegion;
744 if(!windowRegion->CheckError() && !spriteRegion->CheckError())
746 // Purple regions are about to be drawn
747 DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80), windowRegion);
748 DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80), spriteRegion);
750 aWin.Render(aGc, *windowRegion, *spriteRegion); //When not in ChangeTracking mode, both windowRegion and spriteRegion points to the same region
751 aWin.ClearScheduledRegion();
752 aWin.ClearScheduledSpriteRegion();
754 // Green regions have been drawn
755 DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80), windowRegion);
756 DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80), spriteRegion);
760 OomRender(aWin, aGc, aScheduler);
763 clippedSpriteRegion.Close();
764 clippedWindowRegion.Close();
767 void CScreenRedraw::OomRender(CWsWindow& aWin, MWsGraphicsContext& aGc, const TWalkWindowTreeSchedule& aScheduler)
769 const TRegion* windowRegion = &(aScheduler.WindowRegion(aWin));
770 const TRegion* spriteRegion = &(aScheduler.SpriteRegion(aWin));
772 WS_ASSERT_DEBUG(!(windowRegion->IsEmpty() && spriteRegion->IsEmpty()), EWsPanicRegionNull);
774 TRect validWindow(aWin.Abs());
775 validWindow.Intersection(iScreen.RootWindow()->Abs());
777 TRegionFix<1> fallbackRegion(validWindow);
779 if(windowRegion->CheckError())
780 windowRegion = &fallbackRegion;
781 if(spriteRegion->CheckError())
782 spriteRegion = &fallbackRegion;
784 // Purple regions are about to be drawn
785 DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80), windowRegion);
786 DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80), spriteRegion);
788 aWin.Render(aGc, *windowRegion, *spriteRegion); //When not in ChangeTracking mode, both windowRegion and spriteRegion points to the same region
789 aWin.ClearScheduledRegion();
790 aWin.ClearScheduledSpriteRegion();
792 // Green regions have been drawn
793 DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80), windowRegion);
794 DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80), spriteRegion);
798 void CScreenRedraw::DiscardAllSchedules()
802 iTimedDrawRect.Reset();
807 Indicates that a window has moved or changed ordinal position so that the visible regions
808 of all windows needs to be recalculated
810 void CScreenRedraw::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
812 if (iScreen.ChangeTracking() && iScreen.WindowVisibilityNotifier())
815 iRegionUpdateScheduled = ETrue;
817 if(!iScreen.ChangeTracking() && aDefinitelyDirty)
819 iInvalid.Union(*aDefinitelyDirty);
820 // Cyan regions for invalidations caused by this:
821 DEBUG_REGION(TRgb(0x00, 0xFF, 0xFF),TRgb(0x00, 0xFF, 0xFF, 0x20),aDefinitelyDirty);
826 Only used in CHANGETRACKING mode.
827 Windows are scheduled to:
828 - if required, ask client to validate (from lower loop)
829 - render the dirty window region
831 void CScreenRedraw::ScheduleWindow(CWsWindow* aWindow)
833 // Add a scheduled window to the linked list, ignoring duplicates
834 CWsWindow* win = iScheduledWindowList;
835 while (win && win != aWindow)
837 win = win->NextScheduled();
841 aWindow->SetNextScheduled(iScheduledWindowList);
842 iScheduledWindowList = aWindow;
847 void CScreenRedraw::RemoveFromScheduledList(CWsWindow* aWindow)
849 // Search for the window
850 CWsWindow* win = iScheduledWindowList;
851 CWsWindow* previous = NULL;
852 while (win && win != aWindow)
855 win = win->NextScheduled();
859 // Found it, remove from list
860 if (win == iScheduledWindowList)
862 iScheduledWindowList = win->NextScheduled();
866 previous->SetNextScheduled(win->NextScheduled());
871 void CScreenRedraw::RemoveFromTimedDrawList(CWsWindow* aWindow)
875 TInt count(iTimedDrawRect.Count());
879 if (iTimedDrawRect[index].iWindow == aWindow)
881 iTimedDrawRect.Remove(index);
893 Recalculates visible regions and schedules redraws or queues redraw events in response to
896 void CScreenRedraw::RegionUpdate()
898 TBool somethingScheduled = EFalse;
900 if (iVisibilityUpdateScheduled)
902 iVisibilityUpdateScheduled = EFalse;
903 somethingScheduled = ETrue;
904 MWsWindowVisibilityNotifier* const notifier = iScreen.WindowVisibilityNotifier();
907 notifier->SendVisibilityChanges(); // Should result in one callback to SetWindowVisibility for each window subscribing for visible region updates
911 if (iRegionUpdateScheduled)
913 iRegionUpdateScheduled = EFalse;
914 somethingScheduled = ETrue;
915 TWalkWindowTreeUpdateRegions updater(iScreen);
918 for ( TInt idx = iQuickFadeList.Count() - 1; idx >= 0; idx-- )
920 CWsWindow* win = iQuickFadeList[ idx ];
921 const TRegion& quickFadeRegion = win->QuickFadeRegion();
922 //If QuickFadeRegion().IsEmpty() we should remove the window from iQuickFadeList.
923 //And if this window has not been drawn to the screen, then it is not possible to quick fade it.
924 if (quickFadeRegion.IsEmpty() || !win->HasBeenDrawnToScreen())
926 iQuickFadeList.Remove( idx );
931 if (somethingScheduled)
933 TWsPointer::ReLogPointersCurrentWindows();
934 CWsTop::TriggerRedraws(iScreen.RootWindow());
939 To be called by MWsWindowVisibilityNotifier (when running in ChangeTracking mode)
940 in response to MWsWindowVisibilityNotifier::SendVisibilityChanges()
942 void CScreenRedraw::SetWindowVisibility(const MWsWindow& aWindow, const TRegion& aVisibleRegion)
944 CWsWindow& win = static_cast<CWsWindow&>(const_cast<MWsWindow&>(aWindow));
945 WS_ASSERT_DEBUG(win.IsTrackingVisibleRegion(), EWsPanicVisibleRegionTracking);
946 WS_ASSERT_DEBUG(!aVisibleRegion.CheckError(), EWsPanicRegion);
948 if(aVisibleRegion.IsEmpty() && !win.VisibleRegion().IsEmpty())
950 win.ClearVisibleRegion();
952 else if(!aVisibleRegion.IsEmpty() && !aVisibleRegion.CheckError())
954 // Assert that aVisibleRegion is contained by aWin
955 TRect bounds = win.AbsRect();
957 WS_ASSERT_DEBUG(bounds.Contains(aVisibleRegion.BoundingRect().iTl), EWsPanicRegion);
958 WS_ASSERT_DEBUG(bounds.Contains(aVisibleRegion.BoundingRect().iBr), EWsPanicRegion);
959 win.SetVisibleRegion(aVisibleRegion, NULL);
961 else if(aVisibleRegion.CheckError())
963 const TRegionFix<1> fallback(win.Abs());
964 win.SetVisibleRegion(fallback, NULL);
969 To be called by MWsWindowVisibilityNotifier (when running in ChangeTracking mode) when the
970 iVisibleRegion is changed for a window that subscribes for this information. The visible region
971 of the window has not been updated yet at the time of this function call, but the new metrics will be retrieved
972 from MWsWindowVisibilityNotifier (through a call to MWsWindowVisibilityObserver::SetWindowVisibility())
973 and set to each window next time OnAnimation is called.
975 void CScreenRedraw::VisibilityChanged()
977 iVisibilityUpdateScheduled = ETrue;
981 void CScreenRedraw::SetObserver(MWsScreenRedrawObserver* aObserver)
983 iObserver = aObserver;
987 Returns ETrue if an update or animation is scheduled.
989 Note: Now that WSERV surface and composition updates are asynchronous (with NGA)
990 it is possible for this method to return EFalse, even if the last update has
991 not been completed by the rendering pipeline. This is dependant on the configuration
992 and implementation of render stages that make up the rendering pipeline.
994 If in doubt, it is best to call CScreenRedraw::DoRedrawNow(), which will not return
995 untill all updates have been signalled as being complete (note that if signalling
996 is not used, then even this cannot guarantee completion).
998 TBool CScreenRedraw::IsUpdatePending()
1000 if(iScheduled || iAnimating)
1007 Overidding MWsObjectProvider
1009 TAny* CScreenRedraw::ResolveObjectInterface(TUint aTypeId)
1011 TAny* interface = NULL;
1015 case KWsScreenRedraw:
1016 interface = static_cast<MWsScreenRedraw*>(this);
1022 interface = iRenderStages->ResolveObjectInterface(aTypeId);
1028 const TRegion * CScreenRedraw::AnimationRegion() const
1031 return iAnimationRegion;
1036 void CScreenRedraw::UpdateDevice()
1038 //this used to call iScreen->Update, not needed anymore in NGA
1041 TBool CScreenRedraw::IsQuickFadeScheduled( CWsWindow* aWin ) const
1043 const TInt idx = iQuickFadeList.Find( aWin );
1044 return (idx > KErrNotFound);
1047 void CScreenRedraw::ScheduleQuickFade( CWsWindow* aWin )
1049 const TInt idx = iQuickFadeList.Find( aWin );
1050 if ( idx == KErrNotFound )
1051 { // not yet enlisted
1052 if ( KErrNone != iQuickFadeList.Append(aWin) )
1053 { // out of resources?
1054 const TRegion& winReg = aWin->VisibleRegion();
1055 ScheduleRegionUpdate( &winReg );
1060 void CScreenRedraw::RemoveFromQuickFadeList( CWsWindow* aWin )
1062 const TInt idx = iQuickFadeList.Find( aWin );
1063 if ( idx > KErrNotFound )
1065 iQuickFadeList.Remove( idx );
1069 void CScreenRedraw::AcceptFadeRequest( CWsWindow* aWin, TBool aFadeOn )
1073 if (aWin->WinType() != EWinTypeGroup)
1075 ScheduleQuickFade(aWin);
1076 ScheduleRegionUpdate(NULL);
1080 { // fade off, just initiate redraw
1081 if ( !aWin->IsDSAHost() )
1083 AddRedrawRegion(aWin->VisibleRegion() );
1084 ScheduleRegionUpdate(NULL);
1090 #ifdef USE_DEBUG_FRAME_CAPTURE
1091 class TTruncateOverflow : public TDesOverflow
1093 public: //from TDesOverflow
1094 void Overflow(TDes&) {};
1097 void CScreenRedraw::CaptureFrame(const TRegion* aRegion)
1099 MWsScreenDevice* screenDevice = static_cast<MWsScreen&>(iScreen).ObjectInterface<MWsScreenDevice>();
1100 WS_ASSERT_ALWAYS(screenDevice, EWsPanicScreenDeviceMissing);
1101 const TSize screenSize(screenDevice->SizeInPixels());
1102 const TDisplayMode screenDisplayMode(screenDevice->DisplayMode());
1104 //copy the whole screen
1105 TRAPD(err, SetupFrameCaptureResourcesL(screenSize, screenDisplayMode);
1106 screenDevice->CopyScreenToBitmapL(iDebugBitmap, TRect(screenSize)));
1111 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, _L("CScreenRedraw::CaptureFrame(): Failed to create resources for screen capture"));
1115 //remove what's not part of the region
1116 iDebugBitmapGc->Activate(iDebugBitmapDevice);
1118 if(aRegion && !aRegion->IsEmpty() && !aRegion->CheckError())
1120 RWsRegion inverseRegion;
1121 inverseRegion.AddRect(TRect(screenSize));
1122 const TRect* rectList = aRegion->RectangleList();
1123 for(TInt i = aRegion->Count() - 1; i >= 0; i--)
1125 inverseRegion.SubRect(rectList[i]);
1127 if(!inverseRegion.CheckError())
1129 iDebugBitmapGc->SetDrawMode(CGraphicsContext::EDrawModeWriteAlpha);
1130 iDebugBitmapGc->SetClippingRegion(inverseRegion);
1131 iDebugBitmapGc->SetBrushColor(TRgb(55, 55, 55, 0));
1132 iDebugBitmapGc->Clear();
1133 iDebugBitmapGc->CancelClippingRegion();
1134 iDebugBitmapGc->SetDrawMode(CGraphicsContext::EDrawModePEN);
1136 inverseRegion.Close();
1140 const TUint timestamp = User::FastCounter();
1142 TTruncateOverflow overflow;
1143 filename.AppendFormat(iFrameCaptureLocation, &overflow);
1144 filename.AppendFormat(_L("frame_%010u.mbm"), &overflow, timestamp);
1145 iDebugBitmap->Save(filename);
1148 LogRegion(filename, _L(" CScreenRedraw::CaptureFrame() "), aRegion);
1151 void CScreenRedraw::SetupFrameCaptureResourcesL(const TSize& aScreenSize, TDisplayMode aScreenDisplayMode)
1153 //make sure the existing bitmap has the correct display mode
1154 if(iDebugBitmap && iDebugBitmap->DisplayMode() != aScreenDisplayMode)
1156 if(iDebugBitmap->SetDisplayMode(aScreenDisplayMode) != KErrNone)
1158 delete iDebugBitmap;
1159 iDebugBitmap = NULL;
1160 delete iDebugBitmapDevice;
1161 iDebugBitmapDevice = NULL;
1165 //make sure the existing bitmap has the correct size
1166 if(iDebugBitmap && iDebugBitmap->SizeInPixels() != aScreenSize)
1168 if(iDebugBitmap->Resize(aScreenSize) != KErrNone)
1170 delete iDebugBitmap;
1171 iDebugBitmap = NULL;
1172 delete iDebugBitmapDevice;
1173 iDebugBitmapDevice = NULL;
1177 //make sure the bitmap and bitmap device is created
1180 WS_ASSERT_ALWAYS(!iDebugBitmapDevice, EWsPanicTemp); //this should never occur, they should always be created/destroyed in tandem
1181 CFbsBitmap* bitmap = new(ELeave) CFbsBitmap;
1182 CleanupStack::PushL(bitmap);
1183 User::LeaveIfError(bitmap->Create(aScreenSize, aScreenDisplayMode));
1185 iDebugBitmapDevice = CFbsBitmapDevice::NewL(bitmap);
1186 iDebugBitmap = bitmap;
1187 CleanupStack::Pop(bitmap);
1190 //make sure the gc is created
1193 User::LeaveIfError(iDebugBitmapDevice->CreateContext(iDebugBitmapGc));
1197 void CScreenRedraw::LogRegion(const TDesC& aPrefix, const TDesC& aFunctionName, const TRegion* aRegion)
1202 TBuf<LogTBufSize> log;
1203 TTruncateOverflow overflow;
1204 TInt rectCount = (aRegion == NULL ? 0 : aRegion->Count());
1205 log.AppendFormat(aPrefix, &overflow);
1206 log.AppendFormat(aFunctionName, &overflow);
1207 log.AppendFormat(_L("region [%d,"), &overflow, rectCount);
1210 const TRect* rectangles = aRegion->RectangleList();
1212 for (TInt ii = 0; ii < rectCount; ii++)
1214 TRect current = rectangles[ii];
1215 log.AppendFormat(_L("%S{{%d,%d},{%d,%d}}"), &overflow, &comma,
1216 current.iTl.iX,current.iTl.iY,current.iBr.iX,current.iBr.iY);
1222 log.AppendFormat(_L("NULL"), &overflow);
1224 log.AppendFormat(_L("]"), &overflow);
1225 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1227 #endif //USE_DEBUG_FRAME_CAPTURE