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.
16 #include "walkwindowtree.h"
19 #include "offscreenbitmap.h"
24 TWalkWindowTreeFocusChanged::TWalkWindowTreeFocusChanged(TBool aNewFocusState) :
25 iNewFocusState(aNewFocusState)
29 TBool TWalkWindowTreeFocusChanged::DoIt(CWsWindow *aWin)
31 // Walk all windows that have had their focus state changed
34 aWin->FocusChanged(iNewFocusState);
38 TResumableWalkWindowTreeFindInvalid::TResumableWalkWindowTreeFindInvalid(CWsWindowRedraw** aResult) :
43 TBool TResumableWalkWindowTreeFindInvalid::DoIt(CWsWindow* aWin)
45 // Find a window with an invalid area
48 WS_ASSERT_DEBUG(aWin->WinType()==EWinTypeClient, EWsPanicWindowType);
49 CWsWindowRedraw *redraw=((CWsClientWindow *)aWin)->Redraw();
50 if (redraw->NeedsRedraw()>0)
58 TWalkWindowTreeDisconnect::TWalkWindowTreeDisconnect(RWsTextCursor *aCursor) :
62 TBool TWalkWindowTreeDisconnect::DoIt(CWsWindow *aWin)
64 // Disconnect a window
67 if (aWin->WinType()==EWinTypeClient)
69 CWsClientWindow *win=(CWsClientWindow *)aWin;
70 win->iRedraw->WindowClosing();
71 win->DeactivateAllSprites();
74 iTextCursor->WindowDisconnected(win);
75 CWsAnim::WindowClosing(win->iAnimList); // Destroy any animated objects attached to this window
76 WsPointer::WindowDisconected(aWin);
81 win->iFlags&=~EFlagActive;
82 win->ResetHiddenFlag();
87 TWalkWindowTreeRegionBase::TWalkWindowTreeRegionBase(RWsRegion *aRegion, TTranslucentBehaviour aTranslucentBehaviour) :
88 iTranslucentBehaviour(aTranslucentBehaviour), iRegion(aRegion), iSubRegion(NULL)
91 TBool TWalkWindowTreeRegionBase::DoIt(CWsWindow *aWin)
93 if (aWin->IsVisible())
96 if (aWin->WinType()!=EWinTypeRoot)
99 switch(iTranslucentBehaviour)
101 case EDontWalkTranslucent:
102 static_cast<CWsClientWindow *>(aWin)->SetClippedBaseArea(tmp);
103 iRegion->SubRegion(tmp,iSubRegion);
105 case EWalkTranslucent:
106 static_cast<CWsClientWindow *>(aWin)->SetClippedBaseArea(*iSubRegion);
107 iSubRegion->Intersect(*iRegion);
108 if (iSubRegion->Count() > 0)
110 static_cast<CWsClientWindow *>(aWin)->SetOpaqueClippedBaseArea(tmp);
111 iRegion->SubRegion(tmp);
119 iSubRegion->Copy(*iRegion);
121 if (iSubRegion && (iSubRegion->Count()>0 || iSubRegion->CheckError()))
128 return(iRegion->IsEmpty());
130 TBool TWalkWindowTreeRegionBase::DoIt3(CWsWindow*)
133 TWalkWindowTreeSchedule::TWalkWindowTreeSchedule() :
134 TWalkWindowTreeBase(),
139 CWsWindow * TWalkWindowTreeSchedule::HeadWindow() const
144 TWalkWindowTreeScheduleRegions::TWalkWindowTreeScheduleRegions(RWsRegion *aRegion, const TRegion& aTopLayer) :
145 TWalkWindowTreeSchedule(),
147 iTopLayer(aTopLayer),
148 iScheduledRegionsOk(ETrue)
152 // This is similar to TWalkWindowTreeRegionBase::DoIt
153 TBool TWalkWindowTreeScheduleRegions::DoIt(CWsWindow *aWin)
155 WS_ASSERT_DEBUG((aWin != iHead), EWsPanicScheduledRedraw);
156 if (aWin->IsVisible())
158 // Calculate the region we care about for this window:
160 if (aWin->WinType()==EWinTypeRoot)
162 region.Copy(*iRegion);
166 static_cast<CWsClientWindow *>(aWin)->SetClippedBaseArea(region);
167 region.Intersect(*iRegion);
169 // If there is a region we care about, remember the window:
170 // NOTE: Even if there are no redraw segments (ReadyToDraw is false) the window should
171 // be scheduled if it has some animations which will be redrawn via PostDrawWindow (cf def131912)
172 if (!region.IsEmpty() && (aWin->ReadyToDraw() || aWin->HasAnimation() || aWin->HasSprite()) )
174 // Add window to linked list:
175 aWin->SetNextScheduled(iHead);
177 // Set the window scheduled region to something appropriate:
178 if (iScheduledRegionsOk)
180 if (region.CheckError())
182 iScheduledRegionsOk = EFalse;
186 iScheduledRegionsOk = aWin->SetScheduledRegion(region);
190 if (aWin->WinType()!=EWinTypeRoot)
192 // Remove the opaque part from our working region:
193 STACK_REGION opaqueRegion;
194 static_cast<CWsClientWindow *>(aWin)->SetOpaqueClippedBaseArea(opaqueRegion);
195 iRegion->SubRegion(opaqueRegion);
196 opaqueRegion.Close();
198 // Where we were drawing transparent and doing top layer only, remove
200 if (!iTopLayer.IsEmpty())
202 region.Intersect(iTopLayer);
203 iRegion->SubRegion(region);
209 return(iRegion->IsEmpty() || !iScheduledRegionsOk);
212 const TRegion * TWalkWindowTreeScheduleRegions::Region(const CWsWindow* aWin) const
214 WS_ASSERT_DEBUG(iScheduledRegionsOk, EWsPanicScheduledRedraw);
215 return aWin->ScheduledRegion();
218 TBool TWalkWindowTreeScheduleRegions::ScheduledRegionsOk() const
220 return iScheduledRegionsOk;
223 TWalkWindowTreeScheduleFallback::TWalkWindowTreeScheduleFallback(CScreen::CFallbackMap * aFallbackMap) :
224 TWalkWindowTreeSchedule(),
225 iFallbackMap(aFallbackMap)
229 // This is similar to TWalkWindowTreeRegionBase::DoIt
230 TBool TWalkWindowTreeScheduleFallback::DoIt(CWsWindow *aWin)
232 WS_ASSERT_DEBUG((aWin != iHead), EWsPanicScheduledRedraw);
233 if (aWin->IsVisible())
235 if (aWin == aWin->RootWindow())
237 aWin->SetNextScheduled(iHead);
242 TBool addWindow = EFalse;
243 CWsClientWindow* cliWin = static_cast<CWsClientWindow *>(aWin);
244 if (cliWin->IsTranslucent())
246 addWindow = ETrue; // costs more to work out than it is worth
247 const TRegion * opaque = cliWin->GetUserOpaqueRegion();
248 if (opaque && !opaque->CheckError())
249 iFallbackMap->FillRegion(*opaque);
253 addWindow = iFallbackMap->FillRegion(*cliWin->BaseArea());
257 aWin->SetNextScheduled(iHead);
263 return(iFallbackMap->Count() < 1);
266 const TRegion * TWalkWindowTreeScheduleFallback::Region(const CWsWindow* aWin) const
268 if (aWin == aWin->RootWindow())
269 return iFallbackMap->Region();
272 const CWsClientWindow* win = static_cast<const CWsClientWindow *>(aWin);
273 const TRegion* region = win->VisibleRegionIfValid();
275 region = win->BaseArea();
280 TWalkWindowTreeIsObscured::TWalkWindowTreeIsObscured(TBool &aResult) :
286 TBool TWalkWindowTreeIsObscured::DoIt(CWsWindow *aWin)
288 if (!aWin->VisibleRegion().IsEmpty())
296 TWalkWindowTreeSetNonFading::TWalkWindowTreeSetNonFading(TBool aNonFading) :
297 iNonFading(aNonFading)
299 TBool TWalkWindowTreeSetNonFading::DoIt(CWsWindow *aWin)
301 aWin->SetNonFading(iNonFading);
305 TWalkWindowTreeSetFaded::TWalkWindowTreeSetFaded(TBool aFaded,CWsWindowBase* aWin,TUint8 aBlackMap,TUint8 aWhiteMap) :
306 iBlackMap(aBlackMap), iWhiteMap(aWhiteMap), iFaded(aFaded), iGroup(aWin->WinGroup())
310 TBool TWalkWindowTreeSetFaded::DoIt(CWsWindow *aWin)
312 if (aWin->WinGroup()!=iGroup)
314 ((CWsClientWindow*)aWin)->SetFaded(iFaded,iBlackMap,iWhiteMap);
318 TWalkWindowTreePurgeEvents::TWalkWindowTreePurgeEvents()
320 TBool TWalkWindowTreePurgeEvents::DoIt(CWsWindow *aWin)
326 TWalkWindowTreeCalcInvalidGraphics::TWalkWindowTreeCalcInvalidGraphics(RWsRegion *aRegion,TRegion &aDirty,const TArray<TGraphicDrawerId>& aInvalid):
327 TWalkWindowTreeRegionBase(aRegion, EWalkTranslucent),
333 void TWalkWindowTreeCalcInvalidGraphics::DestroyRegions()
344 void TWalkWindowTreeCalcInvalidGraphics::CalcInvalid(CScreen& aScreen)
346 if(aScreen.RootWindow())
348 aScreen.RootWindow()->WalkWindowTree(*this,EWalkChildren);
349 if(iRegion->CheckError())
356 TBool TWalkWindowTreeCalcInvalidGraphics::CreateSubRegion()
358 iSubRegion=new RWsRegion;
359 return iSubRegion!=NULL;
362 TBool TWalkWindowTreeCalcInvalidGraphics::DoIt3(CWsWindow *aWin)
364 if (!iDirty.CheckError() && aWin->Redraw() && aWin->Redraw()->Contains(iInvalid,aWin->VisibleRegion()))
366 STACK_REGION intersection;
367 intersection.Intersection(*iSubRegion,aWin->VisibleRegion());
368 iDirty.Union(intersection);
369 intersection.Close();
371 return iDirty.CheckError();
376 TBool TWalkWindowTreeCheck::DoIt(CWsWindow *aWin)
378 if (aWin->WinType()==EWinTypeRoot)
380 WS_ASSERT_DEBUG(aWin->BaseParent()==NULL, EWsPanicWindowCheck);
381 WS_ASSERT_DEBUG(aWin->NextSibling()==NULL, EWsPanicWindowCheck);
385 WS_ASSERT_DEBUG(aWin->WinType()==EWinTypeClient, EWsPanicWindowCheck);
387 if (aWin->BaseChild())
389 WS_ASSERT_DEBUG(aWin->BaseChild()->BaseParent()==aWin, EWsPanicWindowCheck);
391 if (aWin->NextSibling())
393 WS_ASSERT_DEBUG(aWin->NextSibling()->GetPrevSibling()==aWin, EWsPanicWindowCheck);
398 TBool TWalkWindowTreeFindWithFlag::DoIt(CWsWindow *aWin)
400 if (aWin->iFlags & iFlag)
410 #include "wnredraw.h"
411 TWalkWindowTreeRedrawStoreSize::TWalkWindowTreeRedrawStoreSize() : iTotalSize(0)
415 TBool TWalkWindowTreeRedrawStoreSize::DoIt(CWsWindow *aWin)
417 iTotalSize += aWin->Redraw()->SizeInBytes();
422 TBool TWalkWindowTreeFindByHandle::DoIt(CWsWindow *aWin)
424 if (aWin->ClientHandle() == iHandle)
432 TWalkWindowTreeUpdateRegions::TWalkWindowTreeUpdateRegions(CScreen & aScreen) :
437 void TWalkWindowTreeUpdateRegions::Walk()
439 STACK_REGION floatingSpriteRgn;
440 iScreen.SpriteManager()->CalcFloatingSpriteRgn( floatingSpriteRgn, iScreen.RootWindow()->AbsRect() );
441 iVisible.AddRect(iScreen.RootWindow()->AbsRect());
442 iTop.AddRect(iScreen.RootWindow()->AbsRect());
443 iRemainsOfFadableScreen.AddRect( iScreen.RootWindow()->AbsRect() );
444 iTop.SubRegion(floatingSpriteRgn);
445 iScreen.RootWindow()->WalkWindowTree(*this, EWalkChildren);
448 iRemainsOfFadableScreen.Close();
449 floatingSpriteRgn.Close();
452 TBool TWalkWindowTreeUpdateRegions::DoIt(CWsWindow * aWin)
454 if (aWin->IsVisible() && !iVisible.IsEmpty())
456 // Calculate the region we care about for this window:
457 STACK_REGION newVisibleRegion;
458 STACK_REGION newFadableRegion;
459 if (aWin->WinType()==EWinTypeRoot)
461 newVisibleRegion.Copy(iVisible);
465 static_cast<CWsClientWindow *>(aWin)->SetClippedBaseArea(newVisibleRegion);
466 newVisibleRegion.Intersect(iVisible);
467 if (!aWin->IsTranslucent())
469 iVisible.SubRegion(newVisibleRegion);
474 static_cast<CWsClientWindow *>(aWin)->SetOpaqueClippedBaseArea(opaque);
475 iVisible.SubRegion(opaque);
478 //If the window has been faded calculate what region actually needs fading
479 //(i.e. subtract what has already been faded)
480 if ( aWin->FadeCount() && !aWin->IsNonFading() && aWin->IsVisible() && !iRemainsOfFadableScreen.IsEmpty() )
482 newFadableRegion.Copy( newVisibleRegion );
483 newFadableRegion.Intersect( iRemainsOfFadableScreen );
486 aWin->SetVisibleRegion(newVisibleRegion, &iTop, newFadableRegion);
487 iTop.SubRegion(newVisibleRegion);
488 iRemainsOfFadableScreen.SubRegion( newFadableRegion ); // Subtract the new faded region
489 newFadableRegion.Close();
490 newVisibleRegion.Close();
494 if (!aWin->VisibleRegion().IsEmpty())
496 aWin->ClearVisibleRegion();
502 TWalkWindowTreeScheduleRedraws::TWalkWindowTreeScheduleRedraws():
503 iScheduleRedrawFilter( ERedrawFilterNoFilter )
507 TWalkWindowTreeScheduleRedraws::TWalkWindowTreeScheduleRedraws( TUint32 aFilter ):
508 iScheduleRedrawFilter( aFilter )
511 TBool TWalkWindowTreeScheduleRedraws::DoIt(CWsWindow * aWin)
513 if (aWin->WinType() != EWinTypeClient || static_cast<CWsClientWindow *>(aWin)->HasBeenDrawnToScreen())
515 TBool ban = (iScheduleRedrawFilter & ERedrawFilterOmitDSA) && ( aWin->IsDSAHost() );
518 aWin->Screen()->AddRedrawRegion(aWin->VisibleRegion());
524 TWalkWindowTreeOffsetTransparentRegions::TWalkWindowTreeOffsetTransparentRegions(const TPoint& aOffset) :
529 TBool TWalkWindowTreeOffsetTransparentRegions::DoIt(CWsWindow * aWin)
531 if (aWin != aWin->RootWindow())
532 static_cast<CWsClientWindow *>(aWin)->OffsetUserTransparentRegion(iOffset);
536 TWalkWindowTreeRecalcOpaque::TWalkWindowTreeRecalcOpaque()
540 TBool TWalkWindowTreeRecalcOpaque::DoIt(CWsWindow * aWin)
542 if (aWin != aWin->RootWindow())
543 static_cast<CWsClientWindow *>(aWin)->SetUserOpaqueRegion();
547 TBool TWalkWindowTreeReactivateGcs::DoIt(CWsWindow *aWin)
549 if (aWin != aWin->RootWindow())
550 static_cast<CWsClientWindow *>(aWin)->ReactivateGcs();
554 TWalkWindowTreeScheduleFadeNoRedraw::TWalkWindowTreeScheduleFadeNoRedraw()
557 TBool TWalkWindowTreeScheduleFadeNoRedraw::DoIt(CWsWindow *aWin)
559 aWin->Screen()->ScheduleRegionUpdate( aWin->VisibleRegionIfValid() );