Update contrib.
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"
30 CWsWindowBase::CWsWindowBase(CWsClient* aOwner,WH_HANDLES aType, CScreen* aScreen) : CWsScreenObject(aOwner,aType,aScreen)
34 void CWsWindowBase::ConstructL(CWsWindowBase *parent)
37 iSibling=parent->iChild;
39 SetOrdinalPosition(0);
40 iFadeCount = iParent->iFadeCount;
43 CWsWindowBase *CWsWindowBase::GetPrevSibling() const
45 CWsWindowBase **prev= &iParent->iChild;
46 CWsWindowBase *ret=NULL;
50 prev= &(*prev)->iSibling;
55 CWsWindowBase *CWsWindowBase::LastSibling() const
57 const CWsWindowBase *win;
58 for(win=this;win->iSibling;win=win->iSibling)
60 return (CWsWindowBase*)win;
63 CWsWindowBase *CWsWindowBase::PrevSiblingMultiParent() const
65 CWsWindowBase *win=GetPrevSibling();
68 for(CWsWindowBase *parent=iParent->GetPrevSibling();parent;parent=parent->GetPrevSibling())
69 if ((win=parent->iChild)!=NULL)
70 return(win->LastSibling());
74 // Andy - this is a kind of "next cousin" function
75 CWsWindowBase *CWsWindowBase::NextSiblingMultiParent() const
79 for(CWsWindowBase *parent=iParent->iSibling;parent;parent=parent->iSibling)
81 if (parent->iChild!=NULL)
82 return(parent->iChild);
87 TInt CWsWindowBase::OrdinalPosition(TBool aFull) const
91 OwnerPanic(EWservPanicParentDeleted);
93 CWsWindowBase *win=iParent->iChild;
95 while(iOrdinalPriority<win->iOrdinalPriority)
98 for(count=0;win!=this;count++)
103 /** Removes a window from the list of siblings maintained by its parent window.
105 The iSibling stored inside the window we remove is kept unchanged as it may be needed later.
110 void CWsWindowBase::RemoveFromSiblingList()
114 CWsWindowBase **prev= &iParent->iChild;
115 while ((*prev)!=this)
116 prev= &(*prev)->iSibling;
121 /* This const casts in this function are horrible and need revisiting. */
122 CWsWindowGroup *CWsWindowBase::WinGroup() const
130 CWsWindowBase* win = const_cast<CWsWindowBase*>(this);
131 while(win->WinType()!=EWinTypeGroup)
132 win=win->BaseParent();
133 return static_cast<CWsWindowGroup*>(win);
138 return const_cast<CWsWindowGroup*>(static_cast<const CWsWindowGroup*>(this));
144 TBool CWsWindowBase::CheckOrdinalPositionChange(TInt aPos)
146 // This routine checks to see whether the specified new ordinal position
147 // will causes a change, if so returns ETrue else EFalse.
152 OwnerPanic(EWservPanicParentDeleted);
154 CWsWindowBase *win= iParent->iChild;
155 CWsWindowBase *prev= NULL;
156 while(win==this || (win!=NULL && iOrdinalPriority<win->iOrdinalPriority))
163 else if (win==NULL || (win->iSibling==this && iOrdinalPriority>win->iOrdinalPriority))
165 while(aPos--!=0 && win->iSibling!=NULL && iOrdinalPriority==win->iSibling->iOrdinalPriority)
170 void CWsWindowBase::ChangeWindowPosition(TInt aPos,CWsWindowBase* aNewParent)
172 if (aNewParent != iParent)
174 iScreen->ScheduleRegionUpdate(NULL);
175 TWalkWindowTreeScheduleRedraws wwt;
176 WalkWindowTree(wwt, EWalkChildren);
178 else if (WinType() == EWinTypeClient)
180 CWsClientWindow * cliwin = static_cast<CWsClientWindow*>(this);
181 if (cliwin->IsVisible())
183 iScreen->ScheduleRegionUpdate(NULL);
184 if (cliwin->IsTranslucent())
186 // There is still room for optimization here. These redraws are only required if the window
187 // moved through another window and BOTH of them were transparent, otherwise the visible
188 // region change will sort out the redraws required.
189 TWalkWindowTreeScheduleRedraws wwt;
190 WalkWindowTree(wwt, EWalkChildren);
194 else if (WinType() == EWinTypeGroup)
196 iScreen->ScheduleRegionUpdate(NULL);
197 if (static_cast<CWsWindowGroup*>(this)->HasVisibleTranslucentChild())
199 TWalkWindowTreeScheduleRedraws wwt;
200 WalkWindowTree(wwt, EWalkChildren);
204 RemoveFromSiblingList();
205 CWsWindowBase **prevWinPtr= &aNewParent->iChild;
206 while((*prevWinPtr)!=NULL && iOrdinalPriority<(*prevWinPtr)->iOrdinalPriority)
208 prevWinPtr= &(*prevWinPtr)->iSibling;
210 while(aPos--!=0 && *prevWinPtr!=NULL && iOrdinalPriority==(*prevWinPtr)->iOrdinalPriority)
212 prevWinPtr= &(*prevWinPtr)->iSibling;
214 iSibling=*prevWinPtr;
219 void CWsWindowBase::SetOrdinalPosition(TInt aPos)
221 if (CheckOrdinalPositionChange(aPos))
222 ChangeWindowPosition(aPos,iParent);
225 TEventQueueWalkRet EventPurgeFunc(TAny *aPtr, TWsEvent *aEvent)
227 // Callback function for event queue walk
230 return(((CWsWindowBase *)aPtr)->EventPurgeCheck(aEvent));
233 TEventQueueWalkRet CWsWindowBase::EventPurgeCheck(TWsEvent *aEvent)
235 if (aEvent->Handle()==ClientHandle())
236 return(EEventQueueWalkDeleteEvent);
237 return(EEventQueueWalkOk);
240 void CWsWindowBase::PurgeEvents()
242 iWsOwner->EventQueue()->WalkEventQueue(&EventPurgeFunc,this);
245 void CWsWindowBase::Shutdown()
247 // Destroy a window, disconnects from the window tree and destroys all it's child windows
252 if (iParent!=NULL) // Check it's connected to something
255 for(win=this;win && win->iParent!=(CWsWindowBase *)RootWindow();win=win->iParent)
257 RemoveFromSiblingList();
258 TWalkWindowTreeDisconnect wwt2(win ? ((CWsWindowGroup *)win)->TextCursor() : NULL);
259 WalkWindowTree(wwt2,EWalkChildren); // Disconnect all child windows
262 TWindowServerEvent::RemoveFromSwitchOnEventList(*this);
263 TWindowServerEvent::RemoveFromErrorMessageList(*this);
264 TWindowServerEvent::RemoveFromGroupChangeEventEventList(*this);
265 TWindowServerEvent::RemoveFromFocusChangeEventEventList(*this);
266 TWindowServerEvent::RemoveFromGroupListChangeEventEventList(*this);
267 TWindowServerEvent::RemoveFromModifierChangedEventList(*this);
268 TWindowServerEvent::RemoveFromScreenDeviceChangeEventList(*this);
269 CWsTop::StopWindowGettingOffEvents(this);
272 TBool CWsWindowBase::CommandL(TInt aOpcode, TWsWinCmdUnion &aCmd)
274 // If the command is supported by the window base class process it and return ETrue
275 // if it is not supported return EFalse
283 case EWsWinOpSetOrdinalPosition:
284 SetOrdinalPosition(*aCmd.Int);
286 case EWsWinOpOrdinalPriority:
287 SetReply(iOrdinalPriority);
289 case EWsWinOpOrdinalPosition:
290 SetReply(OrdinalPosition(EFalse));
292 case EWsWinOpFullOrdinalPosition:
293 SetReply(OrdinalPosition(ETrue));
295 case EWsWinOpClientHandle:
296 SetReply(iClientHandle);
301 OwnerPanic(EWservPanicParentDeleted);
303 SetReply(iParent->iClientHandle);
305 case EWsWinOpPrevSibling:
309 OwnerPanic(EWservPanicParentDeleted);
312 for(CWsWindowBase *win=this->GetPrevSibling();win;win=win->GetPrevSibling())
314 if (win->iWsOwner==iWsOwner)
316 reply=win->iClientHandle;
323 case EWsWinOpNextSibling:
326 for(CWsWindowBase *win=this->iSibling;win;win=win->iSibling)
328 if (win->iWsOwner==iWsOwner)
330 reply=win->iClientHandle;
338 SetReply(iChild==NULL ? NULL : iChild->iClientHandle);
340 case EWsWinOpScreenNumber:
341 SetReply(Screen()->ScreenNumber());
343 case EWsWinOpWindowGroupId:
346 OwnerPanic(EWservPanicParentDeleted);
348 SetReply(WinGroup()->Identifier());
350 case EWsWinOpEnableOnEvents:
351 TWindowServerEvent::AddToSwitchOnEventListL(*this, *aCmd.EventControl);
353 case EWsWinOpDisableOnEvents:
354 TWindowServerEvent::RemoveFromSwitchOnEventList(*this);
356 case EWsWinOpEnableErrorMessages:
357 TWindowServerEvent::AddToErrorMessageListL(*this, *aCmd.EventControl);
359 case EWsWinOpDisableErrorMessages:
360 TWindowServerEvent::RemoveFromErrorMessageList(*this);
362 case EWsWinOpEnableModifierChangedEvents:
363 TWindowServerEvent::AddToModifierChangedEventListL(*this, aCmd.EnableModifierChangedEvents->modifierMask,
364 aCmd.EnableModifierChangedEvents->circumstances);
366 case EWsWinOpDisableModifierChangedEvents:
367 TWindowServerEvent::RemoveFromModifierChangedEventList(*this);
369 case EWsWinOpEnableGroupChangeEvents:
370 TWindowServerEvent::AddToGroupChangeEventListL(*this);
372 case EWsWinOpDisableGroupChangeEvents:
373 TWindowServerEvent::RemoveFromGroupChangeEventEventList(*this);
375 case EWsWinOpEnableFocusChangeEvents:
376 TWindowServerEvent::AddToFocusChangeEventListL(*this);
378 case EWsWinOpDisableFocusChangeEvents:
379 TWindowServerEvent::RemoveFromFocusChangeEventEventList(*this);
381 case EWsWinOpEnableGroupListChangeEvents:
382 TWindowServerEvent::AddToGroupListChangeEventListL(*this);
384 case EWsWinOpDisableGroupListChangeEvents:
385 TWindowServerEvent::RemoveFromGroupListChangeEventEventList(*this);
387 case EWsWinOpSetCustomPointerCursor:
388 CWsObject *pointercursor;
389 if ((pointercursor=iWsOwner->HandleToObj(*aCmd.Int, WS_HANDLE_POINTER_CURSOR))==NULL)
390 OwnerPanic(EWservPanicSprite);
391 SetPointerCursor((CWsPointerCursor *)pointercursor);
393 case EWsWinOpSetPointerCursor:
394 SetPointerCursorByIndex(*aCmd.UInt);
396 case EWsWinOpClearPointerCursor:
397 SetPointerCursor(NULL);
399 case EWsWinOpSetNonFading:
401 WS_ASSERT_DEBUG(iScreen, EWsPanicNoScreen);
402 // No fading will occur from a graphical perspective, but the fade counts
403 // are maintained for BC reasons.
404 TWalkWindowTreeSetNonFading wwt(*aCmd.Bool);
405 WalkWindowTree(wwt,EWalkChildren);
406 Screen()->AcceptFadeRequest( reinterpret_cast<CWsWindow*>(this),
412 case EWsWinOpSetFade:
416 OwnerPanic(EWservPanicParentDeleted);
418 WS_ASSERT_DEBUG(iScreen, EWsPanicNoScreen);
422 if (aCmd.SetFaded->UseDefaultMap())
424 iScreen->GetFadingParams(blackMap,whiteMap);
428 aCmd.SetFaded->GetFadingParams(blackMap,whiteMap);
431 if (aCmd.SetFaded->IncludeChildren())
433 TWalkWindowTreeSetFaded wwt(aCmd.SetFaded->Faded(),this,blackMap,whiteMap);
434 WalkWindowTree(wwt,EWalkChildren);
438 if (iWinType==EWinTypeGroup)
439 OwnerPanic(EWservPanicOpcode);
440 static_cast<CWsClientWindow*>(this)->SetFaded(aCmd.SetFaded->Faded(),blackMap,whiteMap);
443 if (CWsTop::IsFadeEnabled())
445 Screen()->AcceptFadeRequest( reinterpret_cast<CWsWindow *> (this),
446 aCmd.SetFaded->Faded(),
448 aCmd.SetFaded->IncludeChildren() );
458 TBool CWsWindowBase::QueueEvent(TInt aEvent) const
461 return(WsOwner()->EventQueue()->QueueEvent(iClientHandle, aEvent));
465 void CWsWindowBase::SetPointerCursorByIndex(TInt aIndex)
467 SetPointerCursor(CWsClient::SystemPointerCursor(aIndex));
470 void CWsWindowBase::SetPointerCursor(CWsPointerCursor *aCursor)
472 CWsPointerCursor *old=iPointerCursor;
473 iPointerCursor=aCursor;
475 iPointerCursor->Open();
476 WsPointer::UpdatePointerCursor();
481 TBool CWsWindowBase::TreeIsObscured() const
484 TWalkWindowTreeIsObscured wwt(result);
485 CONST_CAST(CWsWindowBase *,this)->WalkWindowTree(wwt,EWalkChildren);
489 CEventQueue *CWsWindowBase::EventQueue() const
491 return(iWsOwner->EventQueue());
494 void CWsWindowBase::WalkWindowTree(TWalkWindowTreeBase &aWalkClass,TWalkMode aMode)
496 // Walks windows in a front to back order
498 // If mode is EWalkBehind
499 // call DoIt for all windows that are behind 'this'
500 // else if mode is EWalkChildren
501 // call DoIt for all descendents
502 // else if mode is EWalkChildrenAndBehind
503 // call DoIt for for all descendents and windows behind
508 CWsWindowBase *win=this;
509 CWsWindowBase *end=RootWindow();
510 CWsWindowBase *sibling=win->iSibling;
511 CWsWindowBase *parent=win->iParent;
512 if (aMode!=EWalkBehind)
514 if (aMode==EWalkChildren)
523 start: while(win->iChild!=NULL)
528 sibling=win->iSibling; // De-reference win so it can be destroyed by 'DoIt'
530 if (win->iWinType!=EWinTypeGroup && aWalkClass.DoIt((CWsWindow *)win))
536 /* Walks windows in a front to back order
538 If aResume is EFalse the walk is identical to above.
539 Otherwise iWin is taken as the restart point, (any child windows will have been
542 void CWsWindowBase::WalkWindowTree(TResumableWalkWindowTreeBase& aWalkClass, TWalkMode aMode, TBool aResume)
548 aWalkClass.iWin = this;
549 aWalkClass.iEnd = (aMode == EWalkChildren) ? this : RootWindow();
550 aWalkClass.iParent = aWalkClass.iWin->iParent;
551 if (aMode == EWalkBehind)
553 aWalkClass.iNextChild = aWalkClass.iWin->iSibling;
556 { // ensure walk includes this and its child windows
557 aWalkClass.iNextChild = this;
560 else if (aWalkClass.iWin == aWalkClass.iEnd)
562 return; // walk had already reached end
567 if (aWalkClass.iNextChild != NULL)
568 { // walk down tree to a leaf window
569 aWalkClass.iWin = aWalkClass.iNextChild;
570 while (aWalkClass.iWin->iChild != NULL)
572 aWalkClass.iWin = aWalkClass.iWin->iChild;
577 aWalkClass.iWin = aWalkClass.iParent;
579 // De-reference iWin so it can be destroyed by 'DoIt'
580 aWalkClass.iNextChild = aWalkClass.iWin->iSibling;
581 aWalkClass.iParent = aWalkClass.iWin->iParent;
582 if ( ( aWalkClass.iWin->iWinType != EWinTypeGroup ) && aWalkClass.DoIt(static_cast<CWsWindow *>(aWalkClass.iWin)) )
587 while (aWalkClass.iWin != aWalkClass.iEnd);
593 void CWsWindowBase::CheckTree()
595 TWalkWindowTreeCheck wwt1;
596 WalkWindowTree(wwt1,EWalkChildren);
599 enum {ENullWsHandle=0xFFFFFFFF}; // Events delivered to this handle are thrown away
600 TBool CWsWindowBase::IsClientHandleInUse(TUint32 aHandle)
602 if (aHandle==static_cast<TUint>(ENullWsHandle)) //This value has a special meaning in test code
604 CWsObjectIx* index=iWsOwner->ObjectIndex();
605 const CWsObject* obj;
606 TInt length=index->Length();
608 for (ii=0;ii<length;++ii)
611 if (obj && (obj->Type()==WS_HANDLE_WINDOW || obj->Type()==WS_HANDLE_GROUP_WINDOW))
613 if (STATIC_CAST(const CWsWindowBase*,obj)->ClientHandle()==aHandle)
622 TBool CWsWindowBase::IsDSAHost() const