First public contribution.
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 "windowelementset.h"
32 CWsWindow* CWsWindow::iAccessListRecentEnd = 0;
33 CWsWindow* CWsWindow::iAccessListOldestEnd = 0;
37 #define LOG_WINDOW_REDRAW_START(wswin)
38 #define LOG_WINDOW_REDRAW_END(wswin)
39 #define LOG_WINDOW_ANIM_REDRAW_START(wswin)
40 #define LOG_WINDOW_ANIM_REDRAW_END(wswin)
41 #define LOG_WINDOW_SPRITE_REDRAW_START(aWsWin)
42 #define LOG_WINDOW_SPRITE_REDRAW_END(aWsWin)
46 #define LOG_WINDOW_REDRAW_START(wswin) LogWindowRedrawStart(wswin)
47 #define LOG_WINDOW_REDRAW_END(wswin) LogWindowRedrawEnd(wswin)
48 #define LOG_WINDOW_ANIM_REDRAW_START(wswin) LogWindowAnimRedrawStart(wswin)
49 #define LOG_WINDOW_ANIM_REDRAW_END(wswin) LogWindowAnimRedrawEnd(wswin)
50 #define LOG_WINDOW_SPRITE_REDRAW_START(wswin) LogWindowSpriteRedrawStart(wswin)
51 #define LOG_WINDOW_SPRITE_REDRAW_END(wswin) LogWindowSpriteRedrawEnd(wswin)
53 #include "../debuglog/DEBUGLOG.H"
54 extern CDebugLogBase *wsDebugLog;
56 extern void LogRegion(const TRegion* aRegion)
58 TBuf<LogTBufSize> log;
59 TTruncateOverflow overflow;
60 TInt rectCount = (aRegion == NULL ? 0 : aRegion->Count());
61 log.AppendFormat(_L("Fading region: [%d,"), &overflow, rectCount);
64 const TRect* rectangles = aRegion->RectangleList();
66 for (TInt ii = 0; ii < rectCount; ii++)
68 TRect current = rectangles[ii];
69 log.AppendFormat(_L("%S{{%d,%d},{%d,%d}}"), &overflow, &comma,
70 current.iTl.iX,current.iTl.iY,current.iBr.iX,current.iBr.iY);
76 log.AppendFormat(_L("NULL"), &overflow);
78 log.AppendFormat(_L("]"), &overflow);
79 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
82 extern void LogFadeStart(const CWsWindow* aWsWin)
84 if (wsDebugLog && aWsWin->WsOwner())
86 _LIT(KLogDrawCommandsStart, ">> CWsRedrawMsgWindow::Fade() [%S][app %d] RWindow[%d]");
87 const TDesC& clientName = aWsWin->WsOwner()->Client().FullName();
88 TBuf<LogTBufSize> log;
89 TTruncateOverflow overflow;
90 log.AppendFormat(KLogDrawCommandsStart, &overflow, &clientName, aWsWin->WsOwner()->ConnectionHandle(), aWsWin->LogHandle());
91 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
95 extern void LogFadeRegion(const TRegion* aRegion)
103 extern void LogFadeEnd(const CWsWindow* aWsWin)
105 if (wsDebugLog && aWsWin->WsOwner())
107 _LIT(KLogDrawCommandsEnd, "<< CWsRedrawMsgWindow::Fade() [%S][app %d] RWindow[%d]");
108 const TDesC& clientName = aWsWin->WsOwner()->Client().FullName();
109 TBuf<LogTBufSize> log;
110 TTruncateOverflow overflow;
111 log.AppendFormat(KLogDrawCommandsEnd, &overflow, &clientName, aWsWin->WsOwner()->ConnectionHandle(), aWsWin->LogHandle());
112 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
116 static void LogWindowRedrawStart(const CWsWindow& aWsWin)
120 TBuf<LogTBufSize> log;
121 TTruncateOverflow overflow;
122 if(aWsWin.Type() == WS_HANDLE_ROOT_WINDOW)
124 _LIT(KAnnotateWindowRedrawStart, ">> MWsDrawAnnotationObserver::WindowRedrawStart RootWindow for screen:%d");
125 log.AppendFormat(KAnnotateWindowRedrawStart, &overflow, aWsWin.Screen()->ScreenNumber());
129 _LIT(KAnnotateWindowRedrawStart, ">> MWsDrawAnnotationObserver::WindowRedrawStart [%S][app %d] RWindow[%d]");
130 const TDesC& clientName = aWsWin.WsOwner()->Client().FullName();
131 log.AppendFormat(KAnnotateWindowRedrawStart, &overflow, &clientName, aWsWin.WsOwner()->ConnectionHandle(), aWsWin.LogHandle());
133 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
137 static void LogWindowRedrawEnd(const CWsWindow& aWsWin)
141 TBuf<LogTBufSize> log;
142 TTruncateOverflow overflow;
143 if(aWsWin.Type() == WS_HANDLE_ROOT_WINDOW)
145 _LIT(KAnnotateWindowRedrawEnd, "<< MWsDrawAnnotationObserver::WindowRedrawEnd RootWindow for screen:%d");
146 log.AppendFormat(KAnnotateWindowRedrawEnd, &overflow, aWsWin.Screen()->ScreenNumber());
150 _LIT(KAnnotateWindowRedrawEnd, "<< MWsDrawAnnotationObserver::WindowRedrawEnd [%S][app %d] RWindow[%d]");
151 const TDesC& clientName = aWsWin.WsOwner()->Client().FullName();
152 log.AppendFormat(KAnnotateWindowRedrawEnd, &overflow, &clientName, aWsWin.WsOwner()->ConnectionHandle(), aWsWin.LogHandle());
154 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
158 static void LogWindowAnimRedrawStart(const CWsWindow& aWsWin)
162 _LIT(KAnnotateWindowAnimRedrawStart, " >> MWsDrawAnnotationObserver::WindowAnimRedrawStart [%S][app %d] RWindow[%d]");
163 const TDesC& clientName = aWsWin.WsOwner()->Client().FullName();
164 TBuf<LogTBufSize> log;
165 TTruncateOverflow overflow;
166 log.AppendFormat(KAnnotateWindowAnimRedrawStart, &overflow, &clientName, aWsWin.WsOwner()->ConnectionHandle(), aWsWin.LogHandle());
167 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
171 static void LogWindowAnimRedrawEnd(const CWsWindow& aWsWin)
175 _LIT(KAnnotateWindowAnimRedrawEnd, " << MWsDrawAnnotationObserver::WindowAnimRedrawEnd [%S][app %d] RWindow[%d]");
176 const TDesC& clientName = aWsWin.WsOwner()->Client().FullName();
177 TBuf<LogTBufSize> log;
178 TTruncateOverflow overflow;
179 log.AppendFormat(KAnnotateWindowAnimRedrawEnd, &overflow, &clientName, aWsWin.WsOwner()->ConnectionHandle(), aWsWin.LogHandle());
180 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
184 static void LogWindowSpriteRedrawStart(const CWsWindow& aWsWin)
188 TBuf<LogTBufSize> log;
189 TTruncateOverflow overflow;
190 if(aWsWin.Type() == WS_HANDLE_ROOT_WINDOW)
192 _LIT(KAnnotateSpriteRedrawStart, "<< MWsDrawAnnotationObserver::WindowSpriteRedrawStart RootWindow for screen:%d");
193 log.AppendFormat(KAnnotateSpriteRedrawStart, &overflow, aWsWin.Screen()->ScreenNumber());
197 _LIT(KAnnotateSpriteRedrawStart, " >> MWsDrawAnnotationObserver::WindowSpriteRedrawStart [%S][app %d] RWindow[%d]");
198 const TDesC& clientName = aWsWin.WsOwner()->Client().FullName();
199 log.AppendFormat(KAnnotateSpriteRedrawStart, &overflow, &clientName, aWsWin.WsOwner()->ConnectionHandle(), aWsWin.LogHandle());
201 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
205 static void LogWindowSpriteRedrawEnd(const CWsWindow& aWsWin)
209 TBuf<LogTBufSize> log;
210 TTruncateOverflow overflow;
211 if(aWsWin.Type() == WS_HANDLE_ROOT_WINDOW)
213 _LIT(KAnnotateSpriteRedrawEnd, " << MWsDrawAnnotationObserver::WindowSpriteRedrawEnd RootWindow for screen:%d");
214 log.AppendFormat(KAnnotateSpriteRedrawEnd, &overflow, aWsWin.Screen()->ScreenNumber());
218 _LIT(KAnnotateSpriteRedrawEnd, " << MWsDrawAnnotationObserver::WindowSpriteRedrawEnd [%S][app %d] RWindow[%d]");
219 const TDesC& clientName = aWsWin.WsOwner()->Client().FullName();
220 log.AppendFormat(KAnnotateSpriteRedrawEnd, &overflow, &clientName, aWsWin.WsOwner()->ConnectionHandle(), aWsWin.LogHandle());
222 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
228 extern void AnnotateWindowRedrawStart(const CWsWindow& aWsWin, const TRegion& aRegion)
230 LOG_WINDOW_REDRAW_START(aWsWin);
231 MWsDrawAnnotationObserver* annoObs = aWsWin.Screen()->DrawAnnotationObserver();
234 annoObs->WindowRedrawStart(aWsWin, aRegion);
238 extern void AnnotateWindowRedrawEnd(const CWsWindow& aWsWin)
240 LOG_WINDOW_REDRAW_END(aWsWin);
241 MWsDrawAnnotationObserver* annoObs = aWsWin.Screen()->DrawAnnotationObserver();
244 annoObs->WindowRedrawEnd(aWsWin);
248 extern void AnnotateWindowAnimRedrawStart(const CWsWindow& aWsWin, const CWsAnim& aAnim, const TRegion& aRegion)
250 LOG_WINDOW_ANIM_REDRAW_START(aWsWin);
251 MWsDrawAnnotationObserver* annoObs = aWsWin.Screen()->DrawAnnotationObserver();
254 annoObs->WindowAnimRedrawStart(aAnim, aRegion);
258 extern void AnnotateWindowAnimRedrawEnd(const CWsWindow& aWsWin, const CWsAnim& aAnim)
260 LOG_WINDOW_ANIM_REDRAW_END(aWsWin);
261 MWsDrawAnnotationObserver* annoObs = aWsWin.Screen()->DrawAnnotationObserver();
264 annoObs->WindowAnimRedrawEnd(aAnim);
268 extern void AnnotateSpriteRedrawStart(const CWsWindow& aWsWin, const CWsSpriteBase& aSprite, const TRegion& aRegion)
270 LOG_WINDOW_SPRITE_REDRAW_START(aWsWin);
271 MWsDrawAnnotationObserver* annoObs = aWsWin.Screen()->DrawAnnotationObserver();
274 annoObs->SpriteRedrawStart(aSprite, aRegion);
278 extern void AnnotateSpriteRedrawEnd(const CWsWindow& aWsWin, const CWsSpriteBase& aSprite)
280 LOG_WINDOW_SPRITE_REDRAW_END(aWsWin);
281 MWsDrawAnnotationObserver* annoObs = aWsWin.Screen()->DrawAnnotationObserver();
284 annoObs->SpriteRedrawEnd(aSprite);
288 CWsWindow::CWsWindow(CWsClient* aOwner,WH_HANDLES aType, CScreen* aScreen) : CWsWindowBase(aOwner,aType,aScreen),
290 iDSAs(_FOFF(CWsDirectScreenAccess,iMultipleDSALink))
294 CWsWindow::~CWsWindow()
296 //2-phase destruction, Shutdown() is called before destructor
297 iFadableRegion.Reset();
298 iQuickFadeRegion.Reset();
301 void CWsWindow::Fade(MWsGraphicsContext * aGc, const TRegion& aRegion)
303 if ( NULL != iRedraw )
305 iRedraw->Fade( aGc, aRegion );
309 void CWsWindow::Construct()
311 InsertIntoAccessListOldestEnd();
314 void CWsWindow::Shutdown()
316 TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EWindowClosing, 0, 0, this));
317 RemoveFromAccessList();
318 CWsAnim::WindowClosing(iAnimList); // Destroy any animated objects attached to this window
319 iVisibleRegion.Reset();
320 iScheduledRegion.Reset();
321 iScheduledSpriteRegion.Reset();
322 iDirtyWindowRegion.Reset();
323 iDirtySpriteRegion.Reset();
325 CWsWindowBase::Shutdown();
326 SetPointerCursor(NULL);
329 Screen()->RemoveFromScheduledList(this);
330 Screen()->RemoveFromTimedDrawList(this);
331 Screen()->RemoveFromQuickFadeList(this);
335 // Region and invalid area functions //
338 TRect CWsWindow::FullRect() const
340 return(TRect(iOrigin,iRel.Size()));
347 void CWsWindow::AreaCovered(TRegion &aRegion)
349 aRegion.Copy(WindowArea());
352 void CWsWindow::SetNonFading(TBool aNonFade)
354 const TUint oldFlags = iFlags;
357 iFlags|=EFlagNonFadingWindow;
359 iFlags&=(~EFlagNonFadingWindow);
361 //Schedule window for quickfade if non-fading flag is changed
362 if (!Screen()->ChangeTracking() && CWsTop::IsFadeEnabled() && (oldFlags != iFlags) )
364 Screen()->AcceptFadeRequest(this, !aNonFade);
367 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
368 if (windowTreeObserver && oldFlags!=iFlags)
369 windowTreeObserver->FlagChanged(*this, MWsWindowTreeObserver::ENonFading, aNonFade);
372 TPoint CWsWindow::InquireOffset(TUint32 aHandle) const
375 iWsOwner->HandleToWindow(aHandle,&win2);
376 return(iOrigin-win2->Origin());
379 TDisplayMode CWsWindow::SetRequiredDisplayModeL(TDisplayMode)
381 return DisplayMode();
384 TDisplayMode CWsWindow::DisplayMode() const
386 return iScreen->DisplayMode();
389 void CWsWindow::StatusDump(TDes &aBuf)
391 _LIT(KStatusString1,"CWsWindow[0x%x]RWindow[0x%x,%d],Pos(%d,%d),Size(%d,%d)");
392 _LIT(KStatusString3,",Mode=%d");
393 _LIT(KStatusInvisible,",Inv");
394 aBuf.AppendFormat(KStatusString1,this,iClientHandle,LogHandle(),iRel.iTl.iX,iRel.iTl.iY,Size().iWidth,Size().iHeight);
395 TDisplayMode displayMode = DisplayMode();
396 aBuf.AppendFormat(KStatusString3,(TInt&)displayMode);
398 aBuf.Append(KStatusInvisible);
401 TDblQue<TPointerKeyList> *CWsWindow::PointerKeyList() const
406 TInt CWsWindow::AddDSA(CWsDirectScreenAccess& aDirect)
408 TInt error = KErrNone;
410 if (iDSAs.IsEmpty() && !aDirect.IsRegionTrackingOnly())
412 TRAP(error,iRedraw->SetDsaElementL());
417 iDSAs.AddLast(aDirect);
423 void CWsWindow::RemoveDSA(CWsDirectScreenAccess& aDirect)
425 iDSAs.Remove(aDirect);
428 void CWsWindow::AbortAllDSAs()
430 iScreen->AbortDSAs(RDirectScreenAccess::ETerminateRegion,iDSAs);
433 void CWsWindow::PossibleVisibilityChangedEvent(TBool aForceSend)
437 // coverity[unchecked_value]
438 TWservCrEvent crEvent(TWservCrEvent::EWindowVisibilityChanged, HasBeenDrawnToScreen(), IsVisible()? &iVisibleRegion : NULL, this);
439 TWindowServerEvent::NotifyDrawer(crEvent);
441 if (!(iFlags & EFlagGeneratesVisibilityEvents))
449 if (iVisibleRegion.Count() == 0)
454 {// braces here to avoid gccxml error
455 TInt visibleArea = 0;
456 TInt count = iVisibleRegion.Count();
458 for (ii = 0; ii < count; ii++)
460 visibleArea+= iVisibleRegion[ii].Width() * iVisibleRegion[ii].Height();
463 const TRegion* baseRegion = (static_cast<CWsClientWindow*>(this))->BaseArea();
465 count = baseRegion->Count();
466 for (ii = 0; ii < count; ii++)
468 const TRect& rect = (*baseRegion)[ii];
469 baseArea+= rect.Width() * rect.Height();
472 if (visibleArea == baseArea)
478 goto partiallyvisible;
483 if (aForceSend || !(iFlags & EFlagNotCantBeSeen) || !(iFlags & EFlagCanBeSeen))
485 iFlags |= (EFlagCanBeSeen | EFlagNotCantBeSeen);
486 TWindowServerEvent::SendVisibilityChangedEvents(this, TWsVisibilityChangedEvent::EPartiallyVisible | TWsVisibilityChangedEvent::EFullyVisible);
487 // we have to set EPartiallyVisible too for compatibility reasons
492 if (aForceSend || !(iFlags & EFlagCanBeSeen) || (iFlags & EFlagNotCantBeSeen))
494 iFlags |= EFlagCanBeSeen;
495 iFlags &= ~EFlagNotCantBeSeen;
496 TWindowServerEvent::SendVisibilityChangedEvents(this, TWsVisibilityChangedEvent::EPartiallyVisible);
501 if (aForceSend || (iFlags & EFlagCanBeSeen) || (iFlags & EFlagNotCantBeSeen))
503 iFlags &= ~(EFlagCanBeSeen | EFlagNotCantBeSeen);
504 TWindowServerEvent::SendVisibilityChangedEvents(this, TWsVisibilityChangedEvent::ENotVisible);
509 TPoint CWsWindow::Origin() const
514 TRect CWsWindow::AbsRect() const
519 TSize CWsWindow::Size() const
524 TBool CWsWindow::SetScheduledRegion(const TRegion& aRegion)
526 iScheduledRegion.Copy(aRegion);
527 return !iScheduledRegion.CheckError();
530 void CWsWindow::ClearScheduledRegion()
532 iScheduledRegion.Reset();
535 void CWsWindow::SetFadeBehind(TBool aFade)
537 if (aFade != ((iFlags & EFlagFadeBehind) != 0))
539 iFlags ^= EFlagFadeBehind;
543 void CWsWindow::FocusChanged(TBool aNewFocusState)
545 TRAPD(err,FocusChangedL(aNewFocusState));
547 OwnerPanic(EWservPanicAnimLeave);
550 void CWsWindow::FocusChangedL(TBool aNewFocusState)
552 for (CWsAnim * anim = iAnimList; anim; anim = anim->Next())
554 anim->FocusChanged(aNewFocusState);
558 // Moves a window to the recent end of the accessed list
559 void CWsWindow::Accessed()
561 WS_ASSERT_DEBUG(iAccessListRecentEnd && iAccessListOldestEnd, EWsPanicAccessList);
563 RemoveFromAccessList();
564 InsertIntoAccessListRecentEnd();
567 void CWsWindow::InsertIntoAccessListOldestEnd()
570 iAccessListNext = iAccessListOldestEnd;
572 iAccessListNext->iAccessListPrev = this;
573 iAccessListOldestEnd = this;
574 if (!iAccessListRecentEnd)
575 iAccessListRecentEnd = this;
578 void CWsWindow::InsertIntoAccessListRecentEnd()
581 iAccessListPrev = iAccessListRecentEnd;
583 iAccessListPrev->iAccessListNext = this;
584 iAccessListRecentEnd = this;
585 if (!iAccessListOldestEnd)
586 iAccessListOldestEnd = this;
589 void CWsWindow::RemoveFromAccessList()
591 if (iAccessListOldestEnd == this)
592 iAccessListOldestEnd = iAccessListNext;
593 if (iAccessListRecentEnd == this)
594 iAccessListRecentEnd = iAccessListPrev;
596 iAccessListPrev->iAccessListNext = iAccessListNext;
598 iAccessListNext->iAccessListPrev = iAccessListPrev;
601 TBool CWsWindow::ReleaseMemory(MWsMemoryRelease::TMemoryReleaseLevel aLevel)
603 for (CWsWindow * access = iAccessListOldestEnd; access; access = access->iAccessListNext)
607 if (access->Redraw()->ReleaseMemory(aLevel))
614 const TRegion& CWsWindow::VisibleRegion() const
616 return(iVisibleRegion);
619 const TRegion& CWsWindow::QuickFadeRegion() const
621 return iQuickFadeRegion;
624 const TRegion& CWsWindow::FadableRegion() const
626 return iFadableRegion;
629 TBool CWsWindow::IsDSAHost() const
632 if ( !iDSAs.IsEmpty() )
634 TSglQueIter<CWsDirectScreenAccess> iter( (TSglQueBase&)iDSAs );
636 CWsDirectScreenAccess* dsa;
637 while ( (dsa = iter++) != NULL && !res )
639 res = dsa->IsVisible();
645 void CWsWindow::SendState(MWsWindowTreeObserver& aWindowTreeObserver) const
647 aWindowTreeObserver.NodeCreated(*this, ParentNode());
649 if(iFlags & EFlagActive)
651 aWindowTreeObserver.NodeExtentChanged(*this, FullRect());
652 aWindowTreeObserver.NodeActivated(*this);
657 aWindowTreeObserver.FlagChanged(*this, MWsWindowTreeObserver::EVisible, EFalse);
660 if(IsTrackingVisibleRegion())
662 MWsWindowVisibilityNotifier* windowVisibilityNotifier = Screen()->WindowVisibilityNotifier();
663 if(windowVisibilityNotifier)
664 windowVisibilityNotifier->RegisterWindow(*this);
669 aWindowTreeObserver.FlagChanged(*this, MWsWindowTreeObserver::EAlphaChannelTransparencyEnabled, ETrue);
674 aWindowTreeObserver.FlagChanged(*this, MWsWindowTreeObserver::ENonFading, ETrue);
677 if(IsTopClientWindow() && (iFlags&EFlagScreenDeviceInvalid) )
679 aWindowTreeObserver.FlagChanged(*this, MWsWindowTreeObserver::EScreenDeviceValid, EFalse);
683 iAnimList->SendState(aWindowTreeObserver);
685 CWsWindowBase::SendState(aWindowTreeObserver);
688 void CWsWindow::ResetVisibleRegion()
690 if (!iVisibleRegion.IsEmpty())
692 iVisibleRegion.Reset();
694 iFadableRegion.Reset();
695 iQuickFadeRegion.Reset();
698 void CWsWindow::SetVisibleRegion(const TRegion& aNewRegion, const TRegion* aTop)
700 WS_ASSERT_DEBUG(iScreen, EWsPanicNoScreen);
701 STACK_REGION difference;
702 TBool diffs = EFalse;
704 difference.Copy(iVisibleRegion);
705 difference.SubRegion(aNewRegion);
706 if (!difference.IsEmpty())
711 iScreen->AddRedrawRegion(difference, EFalse);
715 difference.Copy(aNewRegion);
716 if (HasBeenDrawnToScreen())
718 difference.SubRegion(iVisibleRegion);
720 if (!difference.IsEmpty())
723 if(!iScreen->ChangeTracking())
725 //the following code will restart animations
726 STACK_REGION topDiff;
727 topDiff.Copy(difference);
728 WS_ASSERT_DEBUG(aTop,EWsPanicRegion);
729 topDiff.Intersect(*aTop);
730 difference.SubRegion(topDiff);
731 iScreen->AddRedrawRegion(topDiff, EFalse, ERedrawTopOnly);
732 iScreen->AddRedrawRegion(difference, EFalse, ERedrawAll);
737 RestartAnimations(aNewRegion);
743 AbortDsaIfRequired(aNewRegion, aTop);
747 ResetVisibleRegion();
748 iVisibleRegion.Copy(aNewRegion);
749 PossibleVisibilityChangedEvent(EFalse);
751 if (Redraw()->HasElement())
753 WS_ASSERT_DEBUG(WinType()==EWinTypeClient,EWsPanicWindowType);
755 if (WinType()==EWinTypeClient)
757 iScreen->WindowElements().SetVisibleRegion(*static_cast<CWsClientWindow*>(this));
762 // Just because the visible region (screen coordinates) didn't change doesn't
763 // mean the invalid region (window coordinates) didn't change, so we always call this.
764 iRedraw->VisibleRegionChange();
767 //This function sets up the quick fadable region.
768 //It removes anything that cannot be quick faded, and schedules it to be drawn in the normal fashion.
769 void CWsWindow::SetFadeableRegion(const TRegion& aNewFadableRegion, const TRegion& aTop)
771 WS_ASSERT_DEBUG(iScreen, EWsPanicNoScreen);
772 iFadableRegion.Copy(aNewFadableRegion);
774 //Try to figure out if any part of iFadableRegion can be quick faded (i.e. fading applied to
775 //the screen without first having to redraw all visible windows intersecting the region).
776 if ( !iFadableRegion.IsEmpty() && iScreen->IsQuickFadeScheduled(this) )
780 //If a window is semitransparent, then we cannot apply a quickfade to it if
781 //the window below is faded too.
782 iScreen->AddRedrawRegion(iVisibleRegion, EFalse, ERedrawAll);
783 iScreen->RemoveFromQuickFadeList(this);
787 iQuickFadeRegion.Intersection(iFadableRegion, aTop);
789 //Remove any regions not possible to quick fade from iQuickFadeRegion and
790 //schedule these regions for full back-front rendering instead.
791 STACK_REGION nonQuickFadableRegion;
793 for(CWsSpriteBase * sprite = iSpriteList; sprite; sprite = sprite->Next())
795 nonQuickFadableRegion.AddRect(sprite->Rect());
798 for(CWsAnim * anim = iAnimList; anim; anim = anim->Next())
800 nonQuickFadableRegion.AddRect(anim->BestRect());
803 RWsTextCursor* const cursor = CWsTop::CurrentTextCursor();
804 if( cursor && (cursor->Window()==this) && cursor->IsStandardCursorActive() )
806 nonQuickFadableRegion.AddRect(cursor->Rect());
809 //Any regions scheduled for fading but partly or fully covered by transparent windows above them
810 STACK_REGION coveredFadableRegion;
811 coveredFadableRegion.Copy(iFadableRegion);
812 coveredFadableRegion.SubRegion(iQuickFadeRegion);
813 nonQuickFadableRegion.Union(coveredFadableRegion);
814 coveredFadableRegion.Close();
816 nonQuickFadableRegion.Tidy();
818 //Remove any regions not possible to quick fade from iQuickFadeRegion
819 iQuickFadeRegion.SubRegion(nonQuickFadableRegion);
821 if (!nonQuickFadableRegion.CheckError())
823 //Schedule normal drawing (full back to front rendering) for the region not possible to quick fade
824 if (!nonQuickFadableRegion.IsEmpty())
826 iScreen->AddRedrawRegion(nonQuickFadableRegion, EFalse, ERedrawAll);
831 //Schedule normal drawing for the whole iVisibleRegion if the calculations are broken
832 iScreen->AddRedrawRegion(iVisibleRegion, EFalse, ERedrawAll);
834 nonQuickFadableRegion.Close();
839 iQuickFadeRegion.Reset();
843 void CWsWindow::RestartAnimations(const TRegion& aNewRegion)
845 //When not ChangeTracking, restarting is handled by AddRedrawRegion (called from CWsWindow::SetVisibleRegion) and TWalkWindowTreeScheduleRegions
846 WS_ASSERT_DEBUG(iScreen->ChangeTracking(),EWsPanicNoChangetracking);
848 //Restart uncovered window animations
849 for (CWsAnim* anim = iAnimList; anim; anim = anim->Next())
851 if(!iScreen->IsScheduled(EWindowAnim, anim->BestRect(), this) && aNewRegion.Intersects(anim->BestRect()))
853 iScreen->ScheduleAnimation(EWindowAnim, anim->BestRect(), 0, 0, 0, this);
856 //Restart uncovered sprite animations
857 for (CWsSpriteBase* sprite = iSpriteList; sprite; sprite = sprite->Next())
859 if(!iScreen->IsScheduled(ESpriteAnim, sprite->Rect(), sprite->Win()) && aNewRegion.Intersects(sprite->Rect()))
861 iScreen->ScheduleAnimation(ESpriteAnim, sprite->Rect(), 0, 0, 0, sprite->Win());
866 void CWsWindow::ClearVisibleRegion()
869 iScreen->AddRedrawRegion(VisibleRegion(), EFalse);
870 ResetVisibleRegion();
871 PossibleVisibilityChangedEvent(EFalse);
872 if (Redraw()->HasElement())
874 WS_ASSERT_DEBUG(WinType()==EWinTypeClient,EWsPanicWindowType);
876 if (WinType()==EWinTypeClient)
878 iScreen->WindowElements().SetVisibleRegion(*static_cast<CWsClientWindow*>(this));
881 iFlags &= ~EFlagDrawnToScreen;
884 void CWsWindow::AbortDsaIfRequired(const TRegion& aNewRegion, const TRegion* aTop)
886 if (!iDSAs.IsEmpty())
888 // If the top visible region of this window has changed, DSA clients may need
889 // to be sent a DSA abort, as they may be drawing to a different region
890 STACK_REGION newTopVisible;
891 newTopVisible.Copy(aNewRegion); // new visible region
894 newTopVisible.Intersect(*aTop); // area of new visible region not obscured by any other opaque or translucent windows
896 // Build a list of DSA clients that need to be sent a DSA abort
897 TSglQue<CWsDirectScreenAccess> dsaList(_FOFF(CWsDirectScreenAccess,iAbortLink));
898 TSglQueIter<CWsDirectScreenAccess> iter(iDSAs);
899 CWsDirectScreenAccess* dsa;
900 while ((dsa=iter++)!=NULL)
902 if (dsa->IsAbortRequired(newTopVisible))
904 dsaList.AddLast(*dsa);
907 if (!dsaList.IsEmpty())
909 iScreen->AbortDSAs(RDirectScreenAccess::ETerminateRegion, dsaList);
911 newTopVisible.Close();
915 const TRegion* CWsWindow::VisibleRegionIfValid() const
917 return iVisibleRegion.CheckError() ? NULL : &iVisibleRegion;
920 TBool CWsWindow::ReadyToDraw() const
922 return iRedraw->ReadyToDraw();
926 This function draws the region specified
928 void CWsWindow::Render(MWsGraphicsContext& aGc, const TRegion& aWindowRegion, const TRegion& aWindowChildNodeRegion)
930 WS_ASSERT_DEBUG(IsVisible() || (WinType() == EWinTypeRoot), EWsPanicScheduledRedraw);
932 AnnotateWindowRedrawStart(*this, aWindowRegion);
935 iFlags |= EFlagDrawnToScreen;
938 iRedraw->PreDrawWindow(&aGc, aWindowRegion);
939 iRedraw->DrawWindow();
940 iRedraw->PostDrawWindow(&aGc, aWindowChildNodeRegion);
943 void CWsWindow::SetNextScheduled(CWsWindow * aWin)
945 iNextScheduled = aWin;
948 CWsWindow * CWsWindow::NextScheduled() const
950 return iNextScheduled;
953 void CWsWindow::DeactivateAllSprites()
955 CWsSpriteBase * current = iSpriteList;
958 CWsSpriteBase * next = current->Next();
959 current->Deactivate();
964 void CWsWindow::ClipRectToViewport(TRect& aRect) const
966 const CWsWindowBase * win = this;
969 if (win->WinType() != EWinTypeGroup)
971 aRect.Intersection(win->AbsRect());
973 win = win->BaseParent();
977 void CWsWindow::AddDirtyWindowRegion(const TRegion& aRegion)
979 iDirtyWindowRegion.Union(aRegion);
980 if (iDirtyWindowRegion.CheckError())
982 iDirtyWindowRegion.Reset();
983 iDirtyWindowRegion.AddRect(AbsRect()); // fallback to potentially visible part of window
987 const TRegion& CWsWindow::DirtyWindowRegion() const
989 return iDirtyWindowRegion;
992 void CWsWindow::ScheduleDirtyWindowRegion()
994 iDirtyWindowRegion.Tidy();
995 iDirtyWindowRegion.Offset(iOrigin); //convert to screen coords
996 iScheduledRegion.Copy(iDirtyWindowRegion);
997 iDirtyWindowRegion.Reset();
998 if (iScheduledRegion.CheckError())
1000 iScheduledRegion.Reset();
1001 iScheduledRegion.AddRect(AbsRect()); // fallback to potentially visible part of window
1005 void CWsWindow::AddDirtySpriteRegion(const TRegion& aRegion)
1007 iDirtySpriteRegion.Union(aRegion);
1008 if (iDirtySpriteRegion.CheckError())
1010 iDirtySpriteRegion.Reset();
1011 iDirtySpriteRegion.AddRect(AbsRect()); // fallback to potentially visible part of window
1015 const TRegion& CWsWindow::DirtySpriteRegion() const
1017 return iDirtySpriteRegion;
1020 void CWsWindow::ScheduleDirtySpriteRegion()
1022 iDirtySpriteRegion.Tidy();
1023 iDirtySpriteRegion.Offset(iOrigin); //convert to screen coords
1024 iScheduledSpriteRegion.Copy(iDirtySpriteRegion);
1025 iDirtySpriteRegion.Reset();
1026 if (iScheduledSpriteRegion.CheckError())
1028 iScheduledSpriteRegion.Reset();
1029 iScheduledSpriteRegion.AddRect(AbsRect()); // fallback to potentially visible part of window
1033 void CWsWindow::ClearScheduledSpriteRegion()
1035 iScheduledSpriteRegion.Reset();
1039 In ChangeTracking mode, wserv is not maintaining the visible region of windows.
1040 Windows that need to keep track of their visible region in ChangeTracking mode
1041 should call this function.
1043 @param aRegister ETrue if iVisibleRegion should be maintained with accurate information,
1048 void CWsWindow::SetupVisibleRegionTracking(TBool aRegister)
1052 ++iVisibleRegionTrackingCounter;
1053 if (iVisibleRegionTrackingCounter==1)
1055 //If visible region tracking has not been setup, let's do it now.
1056 iScreen->SetupVisibleRegionTracking(*this, ETrue);
1059 else if (iVisibleRegionTrackingCounter>0)
1061 --iVisibleRegionTrackingCounter;
1062 if (iVisibleRegionTrackingCounter==0)
1064 //If aReason was the last reason to track visibility, disable further notifications.
1065 iScreen->SetupVisibleRegionTracking(*this, EFalse);
1073 MWsWindow * CWsWindow::FindChildByHandle(TUint32 aHandle)
1075 TWalkWindowTreeFindByHandle wwt(aHandle);
1076 WalkWindowTree(wwt, EWalkChildren);
1080 TUint32 CWsWindow::Handle() const
1082 return ClientHandle();
1085 MWsScreen * CWsWindow::WsScreen() const
1090 TInt CWsWindow::OrdinalPriority() const
1092 return iOrdinalPriority;