Update contrib.
1 // Copyright (c) 1994-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.
21 #include "walkwindowtree.h"
28 #include "backedupwindow.h"
30 #include "offscreenbitmap.h"
32 CWsWindow* CWsWindow::iAccessListRecentEnd = 0;
33 CWsWindow* CWsWindow::iAccessListOldestEnd = 0;
35 CWsWindow::CWsWindow(CWsClient* aOwner,WH_HANDLES aType, CScreen* aScreen) : CWsWindowBase(aOwner,aType,aScreen),
37 iDSAs(_FOFF(CWsDirectScreenAccess,iMultipleDSALink))
41 CWsWindow::~CWsWindow()
43 iFadableRegion.Close();
46 void CWsWindow::Construct()
48 InsertIntoAccessListOldestEnd();
51 void CWsWindow::Shutdown()
53 TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EWindowClosing, 0, 0, this));
54 RemoveFromAccessList();
55 CWsAnim::WindowClosing(iAnimList); // Destroy any animated objects attached to this window
56 iVisibleRegion.Close();
57 iScheduledRegion.Close();
59 CWsWindowBase::Shutdown();
60 SetPointerCursor(NULL);
65 // Region and invalid area functions //
68 TRect CWsWindow::FullRect() const
70 return(TRect(iOrigin,iRel.Size()));
77 void CWsWindow::SetNonFading(TBool aNonFade)
80 iFlags|=EFlagNonFadingWindow;
82 iFlags&=(~EFlagNonFadingWindow);
85 TPoint CWsWindow::InquireOffset(TUint32 aHandle) const
88 iWsOwner->HandleToWindow(aHandle,&win2);
89 return(iOrigin-win2->Origin());
92 TDisplayMode CWsWindow::DisplayMode() const
94 return iScreen->DisplayMode();
97 void CWsWindow::StatusDump(TDes &aBuf)
99 _LIT(KStatusString1,"CWsWindow[0x%x]RWindow[0x%x,%d],Pos(%d,%d),Size(%d,%d)");
100 _LIT(KStatusString3,",Mode=%d");
101 _LIT(KStatusInvisible,",Inv");
102 aBuf.AppendFormat(KStatusString1,this,iClientHandle,LogHandle(),iRel.iTl.iX,iRel.iTl.iY,Size().iWidth,Size().iHeight);
103 aBuf.AppendFormat(KStatusString3,(TInt)iScreen->DisplayMode());
105 aBuf.Append(KStatusInvisible);
108 TDblQue<TPointerKeyList> *CWsWindow::PointerKeyList() const
113 void CWsWindow::AbortAllDSAs()
115 iScreen->AbortDSAs(RDirectScreenAccess::ETerminateRegion,iDSAs);
118 void CWsWindow::PossibleVisibilityChangedEvent(TBool aForceSend)
122 TWservCrEvent crEvent(TWservCrEvent::EWindowVisibilityChanged, HasBeenDrawnToScreen(), IsVisible()? &iVisibleRegion : NULL, this);
123 TWindowServerEvent::NotifyDrawer(crEvent);
125 if (!(iFlags & EFlagGeneratesVisibilityEvents))
133 if (iVisibleRegion.Count() == 0)
138 {// braces here to avoid gccxml error
139 TInt visibleArea = 0;
140 TInt count = iVisibleRegion.Count();
142 for (ii = 0; ii < count; ii++)
144 visibleArea+= iVisibleRegion[ii].Width() * iVisibleRegion[ii].Height();
147 const TRegion* baseRegion = (static_cast<CWsClientWindow*>(this))->BaseArea();
149 count = baseRegion->Count();
150 for (ii = 0; ii < count; ii++)
152 const TRect& rect = (*baseRegion)[ii];
153 baseArea+= rect.Width() * rect.Height();
156 if (visibleArea == baseArea)
162 goto partiallyvisible;
167 if (aForceSend || !(iFlags & EFlagNotCantBeSeen) || !(iFlags & EFlagCanBeSeen))
169 iFlags |= (EFlagCanBeSeen | EFlagNotCantBeSeen);
170 TWindowServerEvent::SendVisibilityChangedEvents(this, TWsVisibilityChangedEvent::EPartiallyVisible | TWsVisibilityChangedEvent::EFullyVisible);
171 // we have to set EPartiallyVisible too for compatibility reasons
176 if (aForceSend || !(iFlags & EFlagCanBeSeen) || (iFlags & EFlagNotCantBeSeen))
178 iFlags |= EFlagCanBeSeen;
179 iFlags &= ~EFlagNotCantBeSeen;
180 TWindowServerEvent::SendVisibilityChangedEvents(this, TWsVisibilityChangedEvent::EPartiallyVisible);
185 if (aForceSend || (iFlags & EFlagCanBeSeen) || (iFlags & EFlagNotCantBeSeen))
187 iFlags &= ~(EFlagCanBeSeen | EFlagNotCantBeSeen);
188 TWindowServerEvent::SendVisibilityChangedEvents(this, TWsVisibilityChangedEvent::ENotVisible);
193 TPoint CWsWindow::Origin() const
198 TRect CWsWindow::AbsRect() const
203 TSize CWsWindow::Size() const
208 TBool CWsWindow::SetScheduledRegion(const RWsRegion& aRegion)
210 iScheduledRegion.Copy(aRegion);
211 return !iScheduledRegion.CheckError();
214 void CWsWindow::ClearScheduledRegion()
216 iScheduledRegion.Clear();
219 void CWsWindow::SetFadeBehind(TBool aFade)
221 if (aFade != ((iFlags & EFlagFadeBehind) != 0))
223 iFlags ^= EFlagFadeBehind;
227 void CWsWindow::FocusChanged(TBool aNewFocusState)
229 TRAPD(err,FocusChangedL(aNewFocusState));
231 OwnerPanic(EWservPanicAnimLeave);
234 void CWsWindow::FocusChangedL(TBool aNewFocusState)
236 for (CWsAnim * anim = iAnimList; anim; anim = anim->Next())
238 anim->FocusChanged(aNewFocusState);
242 // Moves a window to the recent end of the accessed list
243 void CWsWindow::Accessed()
245 WS_ASSERT_DEBUG(iAccessListRecentEnd && iAccessListOldestEnd, EWsPanicAccessList);
247 RemoveFromAccessList();
248 InsertIntoAccessListRecentEnd();
251 void CWsWindow::InsertIntoAccessListOldestEnd()
254 iAccessListNext = iAccessListOldestEnd;
256 iAccessListNext->iAccessListPrev = this;
257 iAccessListOldestEnd = this;
258 if (!iAccessListRecentEnd)
259 iAccessListRecentEnd = this;
262 void CWsWindow::InsertIntoAccessListRecentEnd()
265 iAccessListPrev = iAccessListRecentEnd;
267 iAccessListPrev->iAccessListNext = this;
268 iAccessListRecentEnd = this;
269 if (!iAccessListOldestEnd)
270 iAccessListOldestEnd = this;
273 void CWsWindow::RemoveFromAccessList()
275 if (iAccessListOldestEnd == this)
276 iAccessListOldestEnd = iAccessListNext;
277 if (iAccessListRecentEnd == this)
278 iAccessListRecentEnd = iAccessListPrev;
280 iAccessListPrev->iAccessListNext = iAccessListNext;
282 iAccessListNext->iAccessListPrev = iAccessListPrev;
285 TBool CWsWindow::ReleaseMemory(MWsMemoryRelease::TMemoryReleaseLevel aLevel)
287 for (CWsWindow * access = iAccessListOldestEnd; access; access = access->iAccessListNext)
291 if (access->Redraw()->ReleaseMemory(aLevel))
298 const TRegion& CWsWindow::VisibleRegion() const
300 return(iVisibleRegion);
303 TBool CWsWindow::IsDSAHost() const
306 if ( !iDSAs.IsEmpty() )
308 TSglQueIter<CWsDirectScreenAccess> iter( (TSglQueBase&)iDSAs );
310 CWsDirectScreenAccess* dsa;
311 while ( (dsa = iter++) != NULL && !res )
313 res = dsa->IsVisible() || dsa->IsSyncTimeoutPending();
319 void CWsWindow::ResetVisibleRegion()
321 if (!iVisibleRegion.IsEmpty())
323 iVisibleRegion.Clear();
325 iFadableRegion.Clear();
328 void CWsWindow::SetVisibleRegion(const TRegion& aNewRegion, const TRegion* aTop, TRegion& aNewFadableRegion)
330 STACK_REGION difference;
331 TBool diffs = EFalse;
333 difference.Copy(iVisibleRegion);
334 difference.SubRegion(aNewRegion);
335 if (!difference.IsEmpty())
340 // Andy - If this is a client window (what else could it be) we can also subtract the
341 // user defined opaque region before doing this:
342 iScreen->AddRedrawRegion(difference, EFalse);
346 difference.Copy(aNewRegion);
347 if (HasBeenDrawnToScreen())
349 difference.SubRegion(iVisibleRegion);
351 if (!difference.IsEmpty())
354 STACK_REGION topDiff;
355 topDiff.Copy(difference);
356 WS_ASSERT_DEBUG(aTop,EWsPanicRegion);
357 topDiff.Intersect(*aTop);
358 difference.SubRegion(topDiff);
359 iScreen->AddRedrawRegion(topDiff, EFalse, ERedrawTopOnly);
360 iScreen->AddRedrawRegion(difference, EFalse, ERedrawAll);
366 AbortDsaIfRequired(aNewRegion, aTop);
370 ResetVisibleRegion();
371 iVisibleRegion.Copy(aNewRegion);
372 PossibleVisibilityChangedEvent(EFalse);
375 iFadableRegion.Copy( aNewFadableRegion );
377 // Just because the visible region (screen coordinates) didn't change doesn't
378 // mean the invalid region (window coordinates) didn't change, so we always call this.
379 iRedraw->VisibleRegionChange();
382 void CWsWindow::ClearVisibleRegion()
385 iScreen->AddRedrawRegion(VisibleRegion(), EFalse);
386 ResetVisibleRegion();
387 PossibleVisibilityChangedEvent(EFalse);
388 iFlags &= ~EFlagDrawnToScreen;
391 void CWsWindow::AbortDsaIfRequired(const TRegion& aNewRegion, const TRegion* aTop)
393 if (!iDSAs.IsEmpty())
395 // If the top visible region of this window has changed, DSA clients may need
396 // to be sent a DSA abort, as they may be drawing to a different region
397 STACK_REGION newTopVisible;
398 newTopVisible.Copy(aNewRegion); // new visible region
401 newTopVisible.Intersect(*aTop); // area of new visible region not obscured by any other opaque or translucent windows
403 // Build a list of DSA clients that need to be sent a DSA abort
404 TSglQue<CWsDirectScreenAccess> dsaList(_FOFF(CWsDirectScreenAccess,iAbortLink));
405 TSglQueIter<CWsDirectScreenAccess> iter(iDSAs);
406 CWsDirectScreenAccess* dsa;
407 while ((dsa=iter++)!=NULL)
409 if (dsa->IsAbortRequired(newTopVisible))
411 dsaList.AddLast(*dsa);
414 if (!dsaList.IsEmpty())
416 iScreen->AbortDSAs(RDirectScreenAccess::ETerminateRegion, dsaList);
418 newTopVisible.Close();
422 const TRegion* CWsWindow::VisibleRegionIfValid() const
424 return iVisibleRegion.CheckError() ? NULL : &iVisibleRegion;
427 TBool CWsWindow::ReadyToDraw() const
429 return iRedraw->ReadyToDraw();
433 This function draws the region specified
434 Andy - I would be happier if no region was specified here - the GC should incorporate that information
436 void CWsWindow::Render(CFbsBitGc * aGc, const TRegion& aRegion)
438 WS_ASSERT_DEBUG(IsVisible() || (WinType() == EWinTypeRoot), EWsPanicScheduledRedraw);
441 iFlags |= EFlagDrawnToScreen;
443 iRedraw->PreDrawWindow(aGc, aRegion);
444 iRedraw->DrawWindow();
445 iRedraw->PostDrawWindow(aGc);
448 void CWsWindow::SetNextScheduled(CWsWindow * aWin)
450 iNextScheduled = aWin;
453 CWsWindow * CWsWindow::NextScheduled() const
455 return iNextScheduled;
458 void CWsWindow::AddSprite(CWsSpriteBase * aSprite)
460 aSprite->SetNext(iSpriteList);
461 iSpriteList = aSprite;
464 void CWsWindow::RemoveSprite(CWsSpriteBase * aSprite)
466 if (aSprite == iSpriteList)
468 iSpriteList = aSprite->Next();
472 for (CWsSpriteBase * sprite = iSpriteList; sprite; sprite = sprite->Next())
474 if (sprite->Next() == aSprite)
476 sprite->SetNext(aSprite->Next());
483 void CWsWindow::DeactivateAllSprites()
485 CWsSpriteBase * current = iSpriteList;
488 CWsSpriteBase * next = current->Next();
489 current->Deactivate();
494 void CWsWindow::ClipRectToViewport(TRect& aRect) const
496 const CWsWindowBase * win = this;
499 if (win->WinType() != EWinTypeGroup)
501 aRect.Intersection(win->AbsRect());
503 win = win->BaseParent();
507 const TRegion& CWsWindow::FadableRegion() const
509 return iFadableRegion;
515 MWsWindow * CWsWindow::FindChildByHandle(TUint32 aHandle)
517 TWalkWindowTreeFindByHandle wwt(aHandle);
518 WalkWindowTree(wwt, EWalkChildren);
522 TUint32 CWsWindow::Handle() const
524 return ClientHandle();
527 MWsScreen * CWsWindow::WsScreen() const
532 TInt CWsWindow::OrdinalPriority() const
534 return iOrdinalPriority;