First public contribution.
1 // Copyright (c) 1995-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.
14 // Window virtual base class, windows and window groups are derived from this
22 #include "windowgroup.h"
23 #include "walkwindowtree.h"
29 #include "windowelementset.h"
31 CWsWindowBase::CWsWindowBase(CWsClient* aOwner,WH_HANDLES aType, CScreen* aScreen) : CWsScreenObject(aOwner,aType,aScreen)
35 void CWsWindowBase::ConstructL(CWsWindowBase *aParent)
38 iSibling=aParent->iChild;
40 CScreen* screen = aParent->Screen();
41 WS_ASSERT_DEBUG(screen,EWsPanicNoScreen);
42 MWsWindowTreeObserver* const windowTreeObserver = screen->WindowTreeObserver();
43 if (windowTreeObserver)
45 windowTreeObserver->NodeCreated(*this, ParentNode());
46 iBaseWinFlags |= EBaseWinNodeCreated;
48 SetOrdinalPosition(0);
49 iFadeCount = iParent->iFadeCount;
52 CWsWindowBase *CWsWindowBase::GetPrevSibling() const
54 if(iParent == NULL) //RootWindow
57 CWsWindowBase* prev=iParent->iChild;
58 CWsWindowBase *ret=NULL;
67 CWsWindowBase *CWsWindowBase::LastSibling() const
69 const CWsWindowBase *win;
70 for(win=this;win->iSibling;win=win->iSibling)
72 return (CWsWindowBase*)win;
75 CWsWindowBase *CWsWindowBase::PrevSiblingMultiParent() const
77 CWsWindowBase *win=GetPrevSibling();
80 for(CWsWindowBase *parent=iParent->GetPrevSibling();parent;parent=parent->GetPrevSibling())
81 if ((win=parent->iChild)!=NULL)
82 return(win->LastSibling());
86 CWsWindowBase *CWsWindowBase::NextSiblingMultiParent() const
90 for(CWsWindowBase *parent=iParent->iSibling;parent;parent=parent->iSibling)
92 if (parent->iChild!=NULL)
93 return(parent->iChild);
98 TInt CWsWindowBase::OrdinalPosition(TBool aFull) const
102 OwnerPanic(EWservPanicParentDeleted);
104 CWsWindowBase *win=iParent->iChild;
106 while(iOrdinalPriority<win->iOrdinalPriority)
109 for(count=0;win!=this;count++)
114 /** Removes a window from the list of siblings maintained by its parent window.
116 The iSibling stored inside the window we remove is kept unchanged as it may be needed later.
121 void CWsWindowBase::RemoveFromSiblingList()
125 CWsWindowBase **prev= &iParent->iChild;
126 while ((*prev)!=this)
127 prev= &(*prev)->iSibling;
132 CWsWindowGroup *CWsWindowBase::WinGroup() const
134 if (iWinType==EWinTypeClient)
135 return(((CWsClientWindow *)this)->TopClientWindow()->Parent());
136 if (iWinType==EWinTypeGroup)
137 return((CWsWindowGroup *)this);
141 TBool CWsWindowBase::CheckOrdinalPositionChange(TInt aPos)
143 // This routine checks to see whether the specified new ordinal position
144 // will causes a change, if so returns ETrue else EFalse.
147 CWsWindowBase *win= iParent->iChild;
148 CWsWindowBase *prev= NULL;
149 while(win==this || (win!=NULL && iOrdinalPriority<win->iOrdinalPriority))
156 else if (win==NULL || (win->iSibling==this && iOrdinalPriority>win->iOrdinalPriority))
158 while(aPos--!=0 && win->iSibling!=NULL && iOrdinalPriority==win->iSibling->iOrdinalPriority)
163 void CWsWindowBase::ChangeWindowPosition(TInt aPos,CWsWindowBase* aNewParent)
165 TBool changedWindowGroup = EFalse;
166 WS_ASSERT_DEBUG(aNewParent,EWsPanicWindowNull);
167 if (aNewParent != iParent)
169 iScreen->ScheduleRegionUpdate(NULL);
170 TWalkWindowTreeScheduleRedraws wwt;
171 WalkWindowTree(wwt, EWalkChildren);
172 changedWindowGroup = ETrue;
174 else if (WinType() == EWinTypeClient)
176 CWsClientWindow * cliwin = static_cast<CWsClientWindow*>(this);
177 if (cliwin->IsVisible())
179 iScreen->ScheduleRegionUpdate(NULL);
180 if (cliwin->IsTranslucent())
182 // There is still room for optimization here. These redraws are only required if the window
183 // moved through another window and BOTH of them were transparent, otherwise the visible
184 // region change will sort out the redraws required.
185 TWalkWindowTreeScheduleRedraws wwt;
186 WalkWindowTree(wwt, EWalkChildren);
190 else if (WinType() == EWinTypeGroup)
192 iScreen->ScheduleRegionUpdate(NULL);
193 if (static_cast<CWsWindowGroup*>(this)->HasVisibleTranslucentChild())
195 TWalkWindowTreeScheduleRedraws wwt;
196 WalkWindowTree(wwt, EWalkChildren);
200 RemoveFromSiblingList();
201 CWsWindowBase **prevWinPtr= &aNewParent->iChild;
202 while((*prevWinPtr)!=NULL && iOrdinalPriority<(*prevWinPtr)->iOrdinalPriority)
204 prevWinPtr= &(*prevWinPtr)->iSibling;
206 while(aPos--!=0 && *prevWinPtr!=NULL && iOrdinalPriority==(*prevWinPtr)->iOrdinalPriority)
208 prevWinPtr= &(*prevWinPtr)->iSibling;
210 iSibling=*prevWinPtr;
214 Screen()->WindowElements().SortByZOrder();
216 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
217 if (windowTreeObserver)
219 if(changedWindowGroup && (WinType() == EWinTypeClient))
221 windowTreeObserver->MovedToWindowGroup(*this, *(this->WinGroup()));
223 else if(!changedWindowGroup)
225 windowTreeObserver->SiblingOrderChanged(*this, OrdinalPosition(ETrue));
227 else if(changedWindowGroup)
229 OwnerPanic(EWservPanicInvalidParameter); //Should be impossible to end up here as only WinType() EWinTypeClient
230 } //and EWinTypeGroup can be moved to another windowgroup.
231 } //@see RWindowBase::MoveToGroup
234 void CWsWindowBase::SetOrdinalPosition(TInt aPos)
236 if (CheckOrdinalPositionChange(aPos))
237 ChangeWindowPosition(aPos,iParent);
240 TEventQueueWalkRet EventPurgeFunc(TAny *aPtr, TWsEvent *aEvent)
242 // Callback function for event queue walk
245 return(((CWsWindowBase *)aPtr)->EventPurgeCheck(aEvent));
248 TEventQueueWalkRet CWsWindowBase::EventPurgeCheck(TWsEvent *aEvent)
250 if (aEvent->Handle()==ClientHandle())
251 return(EEventQueueWalkDeleteEvent);
252 return(EEventQueueWalkOk);
255 void CWsWindowBase::PurgeEvents()
257 iWsOwner->EventQueue()->WalkEventQueue(&EventPurgeFunc,this);
260 void CWsWindowBase::Shutdown()
262 // Destroy a window, disconnects from the window tree and destroys all it's child windows
267 if (iParent!=NULL) // Check it's connected to something
270 for(win=this;win && win->iParent!=(CWsWindowBase *)RootWindow();win=win->iParent)
272 RemoveFromSiblingList();
273 TWalkWindowTreeDisconnect wwt2(win ? ((CWsWindowGroup *)win)->TextCursor() : NULL);
274 WalkWindowTree(wwt2,EWalkChildren); // Disconnect all child windows
277 TWindowServerEvent::RemoveFromSwitchOnEventList(*this);
278 TWindowServerEvent::RemoveFromErrorMessageList(*this);
279 TWindowServerEvent::RemoveFromGroupChangeEventEventList(*this);
280 TWindowServerEvent::RemoveFromFocusChangeEventEventList(*this);
281 TWindowServerEvent::RemoveFromGroupListChangeEventEventList(*this);
282 TWindowServerEvent::RemoveFromModifierChangedEventList(*this);
283 TWindowServerEvent::RemoveFromScreenDeviceChangeEventList(*this);
284 CWsTop::StopWindowGettingOffEvents(this);
287 TBool CWsWindowBase::CommandL(TInt aOpcode, TWsWinCmdUnion &aCmd)
289 // If the command is supported by the window base class process it and return ETrue
290 // if it is not supported return EFalse
300 case EWsWinOpSetOrdinalPosition:
301 SetOrdinalPosition(*aCmd.Int);
303 case EWsWinOpOrdinalPriority:
304 SetReply(iOrdinalPriority);
306 case EWsWinOpOrdinalPosition:
307 SetReply(OrdinalPosition(EFalse));
309 case EWsWinOpFullOrdinalPosition:
310 SetReply(OrdinalPosition(ETrue));
312 case EWsWinOpClientHandle:
313 SetReply(iClientHandle);
318 OwnerPanic(EWservPanicParentDeleted);
320 SetReply(iParent->iClientHandle);
322 case EWsWinOpPrevSibling:
326 OwnerPanic(EWservPanicParentDeleted);
329 for(CWsWindowBase *win=this->GetPrevSibling();win;win=win->GetPrevSibling())
331 if (win->iWsOwner==iWsOwner)
333 reply=win->iClientHandle;
340 case EWsWinOpNextSibling:
343 for(CWsWindowBase *win=this->iSibling;win;win=win->iSibling)
345 if (win->iWsOwner==iWsOwner)
347 reply=win->iClientHandle;
355 SetReply(iChild==NULL ? NULL : iChild->iClientHandle);
357 case EWsWinOpScreenNumber:
358 SetReply(Screen()->ScreenNumber());
360 case EWsWinOpWindowGroupId:
363 CWsWindowGroup *wg=WinGroup();
366 reply=wg->Identifier();
371 case EWsWinOpEnableOnEvents:
373 const TEventControl circumstances = *aCmd.EventControl;
374 TWindowServerEvent::AddToSwitchOnEventListL(*this, circumstances);
375 if (iScreen->ChangeTracking())
377 if(circumstances & EEventControlOnlyWhenVisible)
379 TWalkWindowTreeSetupVisibleRegionTracking wwt(ETrue);
380 WalkWindowTree(wwt, EWalkChildren);
385 case EWsWinOpDisableOnEvents:
387 TWindowServerEvent::RemoveFromSwitchOnEventList(*this);
388 if (iScreen->ChangeTracking())
390 TWalkWindowTreeSetupVisibleRegionTracking wwt(EFalse);
391 WalkWindowTree(wwt, EWalkChildren);
395 case EWsWinOpEnableErrorMessages:
397 const TEventControl circumstances = *aCmd.EventControl;
398 TWindowServerEvent::AddToErrorMessageListL(*this, circumstances);
399 if (iScreen->ChangeTracking())
401 if(circumstances & EEventControlOnlyWhenVisible)
403 TWalkWindowTreeSetupVisibleRegionTracking wwt(ETrue);
404 WalkWindowTree(wwt, EWalkChildren);
409 case EWsWinOpDisableErrorMessages:
411 TWindowServerEvent::RemoveFromErrorMessageList(*this);
412 if (iScreen->ChangeTracking())
414 TWalkWindowTreeSetupVisibleRegionTracking wwt(EFalse);
415 WalkWindowTree(wwt, EWalkChildren);
419 case EWsWinOpEnableModifierChangedEvents:
421 const TInt modifierMask = aCmd.EnableModifierChangedEvents->modifierMask;
422 const TEventControl circumstances = aCmd.EnableModifierChangedEvents->circumstances;
423 TWindowServerEvent::AddToModifierChangedEventListL(*this, modifierMask, circumstances);
424 if (iScreen->ChangeTracking())
426 if(circumstances & EEventControlOnlyWhenVisible)
428 TWalkWindowTreeSetupVisibleRegionTracking wwt(ETrue);
429 WalkWindowTree(wwt, EWalkChildren);
434 case EWsWinOpDisableModifierChangedEvents:
436 TWindowServerEvent::RemoveFromModifierChangedEventList(*this);
437 if (iScreen->ChangeTracking())
439 TWalkWindowTreeSetupVisibleRegionTracking wwt(EFalse);
440 WalkWindowTree(wwt, EWalkChildren);
444 case EWsWinOpEnableGroupChangeEvents:
445 TWindowServerEvent::AddToGroupChangeEventListL(*this);
447 case EWsWinOpDisableGroupChangeEvents:
448 TWindowServerEvent::RemoveFromGroupChangeEventEventList(*this);
450 case EWsWinOpEnableFocusChangeEvents:
451 TWindowServerEvent::AddToFocusChangeEventListL(*this);
453 case EWsWinOpDisableFocusChangeEvents:
454 TWindowServerEvent::RemoveFromFocusChangeEventEventList(*this);
456 case EWsWinOpEnableGroupListChangeEvents:
457 TWindowServerEvent::AddToGroupListChangeEventListL(*this);
459 case EWsWinOpDisableGroupListChangeEvents:
460 TWindowServerEvent::RemoveFromGroupListChangeEventEventList(*this);
462 case EWsWinOpSetCustomPointerCursor:
463 CWsObject *pointercursor;
464 if ((pointercursor=iWsOwner->HandleToObj(*aCmd.Int, WS_HANDLE_POINTER_CURSOR))==NULL)
465 OwnerPanic(EWservPanicSprite);
466 SetPointerCursor((CWsPointerCursor *)pointercursor);
468 case EWsWinOpSetPointerCursor:
469 SetPointerCursorByIndex(*aCmd.UInt);
471 case EWsWinOpClearPointerCursor:
472 SetPointerCursor(NULL);
474 case EWsWinOpSetNonFading:
476 WS_ASSERT_DEBUG(iScreen, EWsPanicNoScreen);
477 // No fading will occur from a graphical perspective, but the fade counts
478 // are maintained for BC reasons.
479 TWalkWindowTreeSetNonFading wwt(*aCmd.Bool);
480 WalkWindowTree(wwt,EWalkChildren);
483 case EWsWinOpSetFade:
485 WS_ASSERT_DEBUG(iScreen, EWsPanicNoScreen);
489 if (aCmd.SetFaded->UseDefaultMap())
491 iScreen->GetFadingParams(blackMap,whiteMap);
495 aCmd.SetFaded->GetFadingParams(blackMap,whiteMap);
498 if (aCmd.SetFaded->IncludeChildren())
500 TWalkWindowTreeSetFaded wwt(aCmd.SetFaded->Faded(),this,blackMap,whiteMap);
501 WalkWindowTree(wwt,EWalkChildren);
505 if (iWinType==EWinTypeGroup)
506 OwnerPanic(EWservPanicOpcode);
508 const TBool KNotifyObserver = ETrue; //yes please
509 const TBool KFaded = aCmd.SetFaded->Faded();
510 static_cast<CWsClientWindow*>(this)->SetFaded(KFaded, blackMap, whiteMap, KNotifyObserver);
514 case EWsWinOpEnableAdvancedPointers:
517 // Must call this BEFORE activating the window.
518 iBaseWinFlags |= EBaseWinAdvancedPointersEnabled;
522 // Called after activation, so panic the client.
523 OwnerPanic(EWservPanicUnableToEnableAdvPointer);
526 case EWsWinOpSetSurfaceTransparency:
527 RDebug::Printf("[Bug 3343] OpCode EWsWinOpSetSurfaceTransparency not supported.");
535 /** @see MWsWindowTreeNode */
536 MWsWindowTreeNode::TType CWsWindowBase::NodeType() const
538 return static_cast<MWsWindowTreeNode::TType>(iWinType); //TWinType is a subset of MWsWindowTreeNode::TType
541 /** @see MWsWindowTreeNode */
542 const MWsWindow* CWsWindowBase::Window() const
544 return (iWinType!=EWinTypeGroup) ? (static_cast<const CWsWindow*>(this)) : NULL;
547 /** @see MWsWindowTreeNode */
548 const MWsSprite* CWsWindowBase::Sprite() const
553 /** @see MWsWindowTreeNode */
554 const MWsStandardTextCursor* CWsWindowBase::StandardTextCursor() const
559 /** @see MWsWindowTreeNode */
560 const MWsWindowGroup* CWsWindowBase::WindowGroup() const
562 return static_cast<MWsWindowGroup*>(WinGroup());
565 /** @see MWsWindowTreeNode */
566 const MWsWindowTreeNode* CWsWindowBase::ParentNode() const
571 TBool CWsWindowBase::QueueEvent(TInt aEvent, TInt aIntVal) const
574 return(WsOwner()->EventQueue()->QueueEvent(iClientHandle, aEvent, aIntVal));
578 void CWsWindowBase::SetPointerCursorByIndex(TInt aIndex)
580 SetPointerCursor(CWsClient::SystemPointerCursor(aIndex));
583 void CWsWindowBase::SetPointerCursor(CWsPointerCursor *aCursor)
585 CWsPointerCursor *old=iPointerCursor;
586 iPointerCursor=aCursor;
588 iPointerCursor->Open();
589 TWsPointer::UpdatePointerCursor();
594 TBool CWsWindowBase::TreeIsObscured() const
597 TWalkWindowTreeIsObscured wwt(result);
598 CONST_CAST(CWsWindowBase *,this)->WalkWindowTree(wwt,EWalkChildren);
602 CEventQueue *CWsWindowBase::EventQueue() const
604 CEventQueue* eventQueue = NULL;
606 eventQueue = iWsOwner->EventQueue();
610 TInt CWsWindowBase::Depth() const
613 const CWsWindowBase *win=this;
614 while (win->WinType()!=EWinTypeRoot)
622 void CWsWindowBase::WalkWindowTree(TWalkWindowTreeBase &aWalkClass,TWalkMode aMode)
624 // Walks windows in a front to back order
626 // If mode is EWalkBehind
627 // call DoIt for all windows that are behind 'this'
628 // else if mode is EWalkChildren
629 // call DoIt for all descendents
630 // else if mode is EWalkChildrenAndBehind
631 // call DoIt for for all descendents and windows behind
636 CWsWindowBase *win=this;
637 CWsWindowBase *end=RootWindow();
638 CWsWindowBase *sibling=win->iSibling;
639 CWsWindowBase *parent=win->iParent;
640 if (aMode!=EWalkBehind)
642 if (aMode==EWalkChildren)
651 start: while(win->iChild!=NULL)
656 sibling=win->iSibling; // De-reference win so it can be destroyed by 'DoIt'
658 if (win->iWinType!=EWinTypeGroup && aWalkClass.DoIt((CWsWindow *)win))
664 /* Walks windows in a front to back order
666 If aResume is EFalse the walk is identical to above.
667 Otherwise iWin is taken as the restart point, (any child windows will have been
670 void CWsWindowBase::WalkWindowTree(TResumableWalkWindowTreeBase& aWalkClass, TWalkMode aMode, TBool aResume)
676 aWalkClass.iWin = this;
677 aWalkClass.iEnd = (aMode == EWalkChildren) ? this : RootWindow();
678 aWalkClass.iParent = aWalkClass.iWin->iParent;
679 if (aMode == EWalkBehind)
681 aWalkClass.iNextChild = aWalkClass.iWin->iSibling;
684 { // ensure walk includes this and its child windows
685 aWalkClass.iNextChild = this;
688 else if (aWalkClass.iWin == aWalkClass.iEnd)
690 return; // walk had already reached end
695 if (aWalkClass.iNextChild != NULL)
696 { // walk down tree to a leaf window
697 aWalkClass.iWin = aWalkClass.iNextChild;
698 while (aWalkClass.iWin->iChild != NULL)
700 aWalkClass.iWin = aWalkClass.iWin->iChild;
705 aWalkClass.iWin = aWalkClass.iParent;
707 // De-reference iWin so it can be destroyed by 'DoIt'
708 aWalkClass.iNextChild = aWalkClass.iWin->iSibling;
709 aWalkClass.iParent = aWalkClass.iWin->iParent;
710 if ( ( aWalkClass.iWin->iWinType != EWinTypeGroup ) && aWalkClass.DoIt(static_cast<CWsWindow *>(aWalkClass.iWin)) )
715 while (aWalkClass.iWin != aWalkClass.iEnd);
719 void CWsWindowBase::WalkWindowTreeBackToFront(TWalkWindowTreeBase &aWalkClass, TWalkModeBackToFront aMode)
721 // Walks windows in a back to front order
723 // If mode is EVisitParentNodesFirst
724 // call DoIt() on each node before walking their child windows.
727 iSibling->WalkWindowTreeBackToFront(aWalkClass,aMode);
729 if(aMode == EVisitParentNodesFirst)
730 aWalkClass.DoIt(static_cast<CWsWindow *>(this));
733 iChild->WalkWindowTreeBackToFront(aWalkClass,aMode);
737 TBool CWsWindowBase::IsDSAHost() const
742 TBool CWsWindowBase::IsActivated() const
747 void CWsWindowBase::AddSprite(CWsSpriteBase * aSprite)
749 aSprite->SetNext(iSpriteList);
750 iSpriteList = aSprite;
753 void CWsWindowBase::RemoveSprite(CWsSpriteBase * aSprite)
755 if (aSprite == iSpriteList)
757 iSpriteList = aSprite->Next();
761 for (CWsSpriteBase * sprite = iSpriteList; sprite; sprite = sprite->Next())
763 if (sprite->Next() == aSprite)
765 sprite->SetNext(aSprite->Next());
772 void CWsWindowBase::SendState(MWsWindowTreeObserver& aWindowTreeObserver) const
776 iSpriteList->SendState(aWindowTreeObserver);
781 void CWsWindowBase::CheckTree()
783 TWalkWindowTreeCheck wwt1;
784 WalkWindowTree(wwt1,EWalkChildren);
787 enum {ENullWsHandle=0xFFFFFFFF}; // Events delivered to this handle are thrown away
788 TBool CWsWindowBase::IsClientHandleInUse(TUint32 aHandle)
790 if (aHandle==static_cast<TUint>(ENullWsHandle)) //This value has a special meaning in test code
792 CWsObjectIx* index=iWsOwner->ObjectIndex();
793 const CWsObject* obj;
794 TInt length=index->Length();
796 for (ii=0;ii<length;++ii)
799 if (obj && (obj->Type()==WS_HANDLE_WINDOW || obj->Type()==WS_HANDLE_GROUP_WINDOW))
801 if (STATIC_CAST(const CWsWindowBase*,obj)->ClientHandle()==aHandle)