Update contrib.
1 // Copyright (c) 1995-2010 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 // Group window sub-class of CWsWindow
23 #include "windowgroup.h"
24 #include "walkwindowtree.h"
30 #include "windowelementset.h"
33 GLREF_D TPtr nullDescriptor;
34 GLREF_D CDebugLogBase* wsDebugLog;
37 TInt CWsWindowGroup::iSkipCount=0;
40 TBool CWsWindowGroup::iEventQueueTest=EFalse; // For stress testing the EventQueue code
42 TInt CWsWindowGroup::iIdentifierCount=1;
43 TBool CWsWindowGroup::iFocusGainPreProcess=EFalse; //'REMOVEFADINGONFOCUSGAIN' flag in INI file
44 RPointerArray< TDblQue<CWsWindowGroup> > CWsWindowGroup::iChains(3);
45 static _LIT_SECURITY_POLICY_C1(KSecurityPolicy_SwEvent,ECapabilitySwEvent);
46 static _LIT_SECURITY_POLICY_C1(KSecurityPolicy_WriteDeviceData,ECapabilityWriteDeviceData);
47 const TInt KArrayMaxGranularity=0x10000000;
49 CWsWindowGroup* CWsWindowGroup::NewL(CWsClient* aOwner, CScreen* aScreen,
50 const TWsClCmdCreateWindowGroup& aCmd)
52 CWsWindowGroup* self = new(ELeave) CWsWindowGroup(aOwner, aScreen);
53 CleanupStack::PushL(self);
54 self->ConstructL(aCmd);
55 CleanupStack::Pop(self);
59 CWsWindowGroup::CWsWindowGroup(CWsClient* aOwner, CScreen* aScreen) : CWsWindowBase(aOwner,WS_HANDLE_GROUP_WINDOW,aScreen)
61 __DECLARE_NAME(_S("CWsWindowGroup"));
62 iWinType=EWinTypeGroup;
65 void CWsWindowGroup::PurgeCapturedKeys()
67 CWsObjectIx& objix=*WsOwner()->ObjectIndex();
68 const TWsObject* ptr=objix.FirstObject();
69 const TWsObject* end=ptr+objix.Length();
70 while(++ptr<end) //Fisrt one should always have a NULL object
72 const CWsObject* obj=ptr->iObject;
74 && ((obj->Type()==WS_HANDLE_CAPTURE_KEY && STATIC_CAST(const CWsCaptureKey*,obj)->WindowGroup()==this)
75 || (obj->Type()==WS_HANDLE_CAPTURE_KEY_UPDOWNS && STATIC_CAST(const CWsCaptureKeyUpsAndDowns*,obj)->WindowGroup()==this)
76 || (obj->Type()==WS_HANDLE_CAPTURE_LONG_KEY && STATIC_CAST(const CWsCaptureLongKey*,obj)->WindowGroup()==this)))
83 CKeyboardRepeat::CancelRepeat(this);
86 void CWsWindowGroup::SwitchToOwningWindow(CWsWindowGroup *aClosingWindow)
88 if (this==CWsTop::FocusWindowGroup())
90 CWsWindowGroup *winGroup=NULL;
92 // First try for an 'owning' window
94 if (iOwningWindowGroup)
96 for(winGroup=RootWindow()->Child();winGroup;winGroup=winGroup->NextSibling())
97 if (winGroup->Identifier()==iOwningWindowGroup && winGroup->iOrdinalPriority==iOrdinalPriority)
101 // If that failed look for the frontmost window belonging to the owner of dying window
103 for(winGroup=RootWindow()->Child();winGroup;winGroup=winGroup->NextSibling())
104 if (winGroup!=this && winGroup->WsOwner()==WsOwner() && winGroup->iOrdinalPriority==iOrdinalPriority)
107 // Next try for the nominated default owning window group
109 winGroup=iScreen->DefaultOwningWindowGroup();
110 if (winGroup && winGroup->iOrdinalPriority==iOrdinalPriority)
112 gotIt: winGroup->SetOrdinalPosition(0,this);
116 ResetFocus(aClosingWindow);
119 CWsWindowGroup::~CWsWindowGroup()
121 MWsWindowTreeObserver* windowTreeObserver = NULL;
122 if (Screen()&& iBaseWinFlags&EBaseWinNodeCreated)
124 windowTreeObserver = Screen()->WindowTreeObserver();
125 if (windowTreeObserver)
127 if( iQueue && (iQueue->Last()!=this) )
129 //This node is part of chain, send notification unless this is the last node in the chain
130 windowTreeObserver->WindowGroupChainBrokenAfter(*this);
132 windowTreeObserver->NodeReleased(*this);
133 iBaseWinFlags &= ~EBaseWinNodeCreated;
136 DisconnectFloatingSprites();
141 _LIT(KWSERVDebugLogGroupWindowId,"Destroying: RWindowGroup[0x%x,%d],Id=%d");
142 buf.Format(KWSERVDebugLogGroupWindowId,iClientHandle,LogHandle(),iIdentifier);
143 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, buf);
145 if (CClick::IsHandler())
146 CClick::OtherEvent(EEventGroupWindowClose,reinterpret_cast<TAny*>(iIdentifier));
149 if (iQueue->Last()!=this)
150 { //Unlink all the children of the window that is being deleted
151 TDblQueIter<CWsWindowGroup> iter(*iQueue);
152 CWsWindowGroup* groupWin;
154 while ((groupWin=iter--)!=this)
156 WS_ASSERT_DEBUG(groupWin!=NULL && groupWin->iQueue==iQueue,EWsPanicGroupWindowChainError);
157 if ( windowTreeObserver && (iQueue->Last()!=groupWin) )
159 //Send notification unless groupWin is the last node in the chain
160 windowTreeObserver->WindowGroupChainBrokenAfter(*groupWin);
162 groupWin->iChainLink.Deque();
163 // coverity[extend_simple_error]
164 groupWin->iQueue=NULL;
167 WS_ASSERT_DEBUG(iQueue->Last()==this,EWsPanicGroupWindowChainError);
168 if(windowTreeObserver && !iQueue->IsEmpty())
170 //If there are chained nodes before this one, notify "chain broken" immediately before this node
171 TDblQueIter<CWsWindowGroup> iter(*iQueue);
172 iter.SetToLast(); //i.e. set to this (see assert above)
173 iter--; //i.e. set to parent
174 CWsWindowGroup* parent = iter;
177 windowTreeObserver->WindowGroupChainBrokenAfter(*parent);
180 TDblQueLinkBase* parentLink=iChainLink.iPrev;
182 if (parentLink->iNext==parentLink->iPrev) //Check to see chain no longer required
184 if (!iQueue->IsEmpty())
185 { //Only the parent is left in queue
186 CWsWindowGroup* parent=iQueue->First();
187 static_cast<TDblQueLink*>(parentLink)->Deque();
188 WS_ASSERT_DEBUG(parent->iQueue==iQueue,EWsPanicGroupWindowChainError);
189 // coverity[extend_simple_error]
195 RemoveAllPriorityKeys();
198 SetPointerCursor(NULL);
199 for(CWsTopClientWindow *win=Child();win;win=win->NextSiblingTop())
203 iScreen->RemoveFromDefaultOwningList(this);
205 CWsWindowBase::Shutdown();
206 TWindowServerEvent::SendGroupChangedEvents();
207 // coverity[extend_simple_error]
208 iClientHandle=0; // To block focus lost events being sent
209 // Decide which window to give focus to if WServ isn't shutting down
210 if (iScreen && !CWsTop::ShuttingDown())
211 SwitchToOwningWindow(this);
213 delete iMessageArray;
216 void CWsWindowGroup::DisconnectFloatingSprites()
218 CWsSpriteBase * current = iSpriteList;
221 CWsSpriteBase * next = current->Next();
222 current->Deactivate();
223 current->DisconnectGroupWin();
228 void CWsWindowGroup::DeleteQueue(TDblQue<CWsWindowGroup>* aQueue)
230 iChains.Remove(iChains.Find(aQueue));
232 if (iChains.Count()==0)
238 void CWsWindowGroup::AdvanceIdentifierCount()
240 if (++iIdentifierCount>EMaxIdentifierCount)
241 iIdentifierCount=1; // so limit it to low value
244 void CWsWindowGroup::ConstructL(const TWsClCmdCreateWindowGroup &aCmd)
247 if (IsClientHandleInUse(aCmd.clientHandle))
249 OwnerPanic(EWservPanicDuplicateHandle);
253 iFlags=EGroupFlagAutoForeground|EGroupFlagMsgQueueNew;
256 iFlags|=EGroupFlagReceivesFocus;
258 iTextCursor.ConstructL(this);
259 iClientHandle=aCmd.clientHandle;
261 if(aCmd.screenDeviceHandle <= 0)
263 //Use primary screen. Client should make sure PrimaryScreenDevice is correct set up immediately after establishing session.
264 iScreenDevice=iWsOwner->PrimaryScreenDevice();
268 //Use the specified screen
269 iScreenDevice=STATIC_CAST(DWsScreenDevice*,iWsOwner->HandleToObj(aCmd.screenDeviceHandle,WS_HANDLE_SCREEN_DEVICE));
272 iScreen = (iScreenDevice) ? iScreenDevice->Screen() : CWsTop::Screen(); //if no screen device use screen 0
274 CWsWindowGroup* parent=NULL;
277 parent=CWsWindowGroup::WindowGroupFromIdentifier(aCmd.parentId);
280 OwnerPanic(EWservPanicWindow);
283 if(parent->Screen() != iScreen)
285 OwnerPanic(EWservPanicWrongScreen);
288 if (parent->iOrdinalPriorityAdjust>0)
290 WS_ASSERT_DEBUG(parent->iQueue==NULL,EWsPanicGroupWindowChainError);
291 parent->iOrdinalPriorityAdjust=0;
292 parent->UpdateOrdinalPriority(ETrue);
294 iOrdinalPriorityBase=parent->iOrdinalPriorityBase;
295 iOrdinalPriority=iOrdinalPriorityBase;
300 AdvanceIdentifierCount(); // Always advance by at least one to stop re-using last id
301 } while (WindowGroupFromIdentifier(iIdentifierCount)); // If current count is in use try again
302 iIdentifier=iIdentifierCount;
304 CWsWindowBase::ConstructL(RootWindow());
305 iScreen->ResetFocus(NULL);
309 TDblQue<CWsWindowGroup>* queue=parent->iQueue;
310 if (queue && queue->Last()!=parent)
311 User::Leave(KErrInUse);
312 if (parent->iWsOwner!=iWsOwner)
314 _LIT_SECURITY_POLICY_S0(securityPolicy,parent->iChildSID);
315 if (!securityPolicy().CheckPolicy(iWsOwner->ClientMessage()))
316 User::Leave(KErrPermissionDenied);
320 queue=new(ELeave) TDblQue<CWsWindowGroup>(_FOFF(CWsWindowGroup,iChainLink));
321 CleanupStack::PushL(queue);
322 User::LeaveIfError(iChains.Append(queue));
323 CleanupStack::Pop(queue);
324 queue->AddFirst(*parent);
325 parent->iQueue=queue;
327 iQueue=queue; //Shouldn't set the queue until after it can leave
328 iChainLink.Enque(&parent->iChainLink);
330 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
331 if (windowTreeObserver)
333 windowTreeObserver->WindowGroupChained(*parent, *this);
336 iMessageArray=new(ELeave) CArrayVarSeg<TWsMessage>(1);
337 if (CClick::IsHandler())
339 TGroupWindowOpenData params;
340 params.iIdentifier=iIdentifier;
341 params.iClient=iWsOwner->ConnectionHandle();
342 params.iNumClientWindowGroups=NumClientWindowGroups()-1; //Don't include this one
343 CClick::OtherEvent(EEventGroupWindowOpen,¶ms);
348 _LIT(KWSERVDebugLogGroupWindowId,"Creating: RWindowGroup[0x%x,%d],Id=%d");
349 buf.Format(KWSERVDebugLogGroupWindowId,iClientHandle,LogHandle(),iIdentifier);
350 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, buf);
354 void CWsWindowGroup::UpdateOrdinalPriority(TBool aDoAdjust)
357 newPri=iOrdinalPriorityBase;
358 if (iWsOwner==CWsTop::FocusWindowGroupOwner())
359 newPri+=iOrdinalPriorityAdjust;
360 CheckCapability(newPri);
361 if (newPri!=iOrdinalPriority)
363 iOrdinalPriority=newPri;
365 SetOrdinalPosition(0);
369 void CWsWindowGroup::SetOrdinalPriority(TInt aPos,TInt aPriority)
373 iOrdinalPriorityBase=aPriority;
374 UpdateOrdinalPriority(EFalse);
378 TDblQueIter<CWsWindowGroup> iter(*iQueue);
379 CWsWindowGroup* group;
380 while ((group=iter++)!=NULL)
382 group->iOrdinalPriorityBase=aPriority;
383 group->iOrdinalPriority=aPriority;
386 SetOrdinalPosition(aPos);
389 void CWsWindowGroup::CommandL(TInt aOpcode, const TAny *aCmdData)
392 // Save root window for performing CheckTree at the end of this func.
393 // When aOpcode is EWsWinOpFree, this object would've been destroyed
394 // and a call to RootWindow() in that case would be impossible
395 CWsRootWindow* rootWindow=RootWindow();
397 // For certain opcodes, check for the 'screen device deleted' condition. If it
398 // has occured for the screen device associated with this group window then
399 // those op-codes are not valid, and the client is panicked.
402 case EWsWinOpEnableScreenChangeEvents:
403 case EWsWinOpAllowChildWindowGroup:
404 case EWsWinOpReceiveFocus:
405 case EWsWinOpAutoForeground:
406 case EWsWinOpSetOrdinalPositionPri:
407 case EWsWinOpSetOrdinalPriorityAdjust:
408 case EWsWinOpCaptureKey:
409 case EWsWinOpCaptureKeyUpsAndDowns:
410 case EWsWinOpCaptureLongKey:
411 case EWsWinOpAddPriorityKey:
412 case EWsWinOpSetTextCursor:
413 case EWsWinOpSetTextCursorClipped:
414 case EWsWinOpSetOwningWindowGroup:
415 case EWsWinOpDefaultOwningWindow:
416 case EWsWinOpSetName:
417 case EWsWinOpDisableKeyClick:
418 case EWsWinOpSendPointerEvent:
419 case EWsWinOpSendAdvancedPointerEvent:
421 if (ScreenDeviceDeleted())
422 OwnerPanic(EWservPanicGroupWinScreenDeviceDeleted);
428 TWsWinCmdUnion pData;
430 if (CWsWindowBase::CommandL(aOpcode,pData)==EFalse)
434 case EWsWinOpAllowChildWindowGroup:
435 iChildSID=*pData.UInt;
437 case EWsWinOpEnableScreenChangeEvents:
438 SetScreenChangeEventStateL(ETrue);
440 case EWsWinOpDisableScreenChangeEvents:
441 SetScreenChangeEventStateL(EFalse);
443 case EWsWinOpReceiveFocus:
444 if (*pData.Bool!=(iFlags&EGroupFlagReceivesFocus))
446 iFlags&=~EGroupFlagReceivesFocus;
448 iFlags|=EGroupFlagReceivesFocus;
449 iScreen->ResetFocus(NULL);
452 case EWsWinOpAutoForeground:
453 iFlags&=~EGroupFlagAutoForeground;
455 iFlags|=EGroupFlagAutoForeground;
457 case EWsWinOpSetOrdinalPositionPri:
458 case EWsWinOpSetOrdinalPositionErr:
460 TInt priority=pData.OrdinalPos->ordinalPriority;
461 TBool hascap = CheckCapability(priority);
462 SetOrdinalPriority(pData.OrdinalPos->pos, priority);
463 if (aOpcode == EWsWinOpSetOrdinalPositionErr)
465 SetReply(hascap?KErrNone:KErrPermissionDenied);
469 case EWsWinOpSetOrdinalPriorityAdjust:
472 iOrdinalPriorityAdjust=*pData.Int;
473 UpdateOrdinalPriority(ETrue);
476 case EWsWinOpCaptureKey:
478 if(!KSecurityPolicy_SwEvent().CheckPolicy(iWsOwner->ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWindowGroup::CaptureKey API")))
480 User::Leave(KErrPermissionDenied);
482 CWsCaptureKey *cKey=new(ELeave) CWsCaptureKey(this);
483 CleanupStack::PushL(cKey);
484 cKey->ConstructL(*pData.CaptureKey);
488 case EWsWinOpCaptureKeyUpsAndDowns:
490 if(!KSecurityPolicy_SwEvent().CheckPolicy(iWsOwner->ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWindowGroup::CaptureKeyUpsAndDowns API")))
492 User::Leave(KErrPermissionDenied);
494 CWsCaptureKeyUpsAndDowns *cKey=new(ELeave) CWsCaptureKeyUpsAndDowns(this);
495 CleanupStack::PushL(cKey);
496 cKey->ConstructL(*pData.CaptureKey);
500 case EWsWinOpCaptureLongKey:
502 if(!KSecurityPolicy_SwEvent().CheckPolicy(iWsOwner->ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWindowGroup::CaptureLongKey API")))
504 User::Leave(KErrPermissionDenied);
506 CWsCaptureLongKey *cKey=new(ELeave) CWsCaptureLongKey(this);
507 CleanupStack::PushL(cKey);
508 cKey->ConstructL(*pData.CaptureLongKey);
512 case EWsWinOpCancelCaptureKey:
513 if (*pData.UInt!=0) // Ignore null handle
515 CWsObject *destroyObj = iWsOwner->HandleToObj(*pData.UInt, WS_HANDLE_CAPTURE_KEY);
518 // Cancel any repeat that is underway for this capture
519 CKeyboardRepeat::CancelRepeat(destroyObj, EFalse);
525 // Attempt to cancel key capture with an incorrect handle
526 OwnerPanic(EWservPanicDestroy);
531 case EWsWinOpCancelCaptureKeyUpsAndDowns:
532 if (*pData.UInt!=0) // Ignore null handle
534 CWsObject *destroyObj = iWsOwner->HandleToObj(*pData.UInt, WS_HANDLE_CAPTURE_KEY_UPDOWNS);
542 // Attempt to cancel ups and downs key capture with an incorrect handle
543 OwnerPanic(EWservPanicDestroy);
548 case EWsWinOpCancelCaptureLongKey:
549 if (*pData.UInt!=0) // Ignore null handle
551 CWsObject *destroyObj = iWsOwner->HandleToObj(*pData.UInt, WS_HANDLE_CAPTURE_LONG_KEY);
554 // Cancel any repeat that is underway for this capture
555 CKeyboardRepeat::CancelRepeat(destroyObj, ETrue);
561 // Attempt to cancel long key capture with an incorrect handle
562 OwnerPanic(EWservPanicDestroy);
567 case EWsWinOpAddPriorityKey:
568 AddPriorityKeyL(pData.PriorityKey->keycode, pData.PriorityKey->modifierMask, pData.PriorityKey->modifiers);
570 case EWsWinOpRemovePriorityKey:
571 RemovePriorityKey(pData.PriorityKey->keycode, pData.PriorityKey->modifierMask, pData.PriorityKey->modifiers);
573 case EWsWinOpSetTextCursor:
574 iTextCursor.SetL(*pData.SetTextCursor, EFalse);
576 case EWsWinOpSetTextCursorClipped:
577 iTextCursor.SetL(*pData.SetTextCursor, ETrue);
579 case EWsWinOpCancelTextCursor:
580 iTextCursor.Cancel();
582 case EWsWinOpSetOwningWindowGroup:
583 iOwningWindowGroup=*pData.Int;
585 case EWsWinOpDefaultOwningWindow:
587 if(KSecurityPolicy_WriteDeviceData().CheckPolicy(iWsOwner->ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWindowGroup::DefaultOwningWindow API")))
589 iScreen->SetDefaultOwningWindow(this);
594 iWsOwner->ReplyGroupName(iGroupName,*pData.Int);
596 case EWsWinOpSetName:
599 const TInt size=*pData.Int;
602 newName=HBufC::NewLC(size);
603 TPtr ptr(newName->Des());
604 iWsOwner->RemoteReadL(ptr,0);
605 CleanupStack::Pop(newName);
607 //Window Group Name is unchanged
608 if (iGroupName && newName && *iGroupName == *newName)
612 else //Window Group Name is changed
616 TWindowServerEvent::SendGroupChangedEvents();
617 if (iScreen && iScreen->ChangeTracking())
619 MWsWindowTreeObserver* const windowTreeObserver = iScreen->WindowTreeObserver();
620 if (windowTreeObserver)
621 windowTreeObserver->AttributeChanged(*this, MWsWindowTreeObserver::EWindowGroupName);
626 case EWsWinOpIdentifier:
627 SetReply(Identifier());
629 case EWsWinOpDisableKeyClick:
631 iFlags|=EGroupFlagDisableKeyClick;
633 iFlags&=~EGroupFlagDisableKeyClick;
634 if (this==CWsTop::FocusWindowGroup())
635 UpdateKeyClickState();
637 case EWsWinOpSendAdvancedPointerEvent:
638 case EWsWinOpSendPointerEvent:
640 TRawEvent eventCopy = *pData.RawEvent;
641 if (TWsPointer::PreProcessClientEvent(eventCopy, aOpcode == EWsWinOpSendAdvancedPointerEvent))
643 if (!TWindowServerEvent::MousePress(eventCopy,this))
645 OwnerPanic(EWservPanicEventType);
650 case EWsWinOpClearChildGroup:
653 TBool fBefore=EFalse;
655 // If there is nothing to clear, return KErrArgument
656 if(iQueue->Last()==this)
658 SetReply(KErrArgument);
661 // fBefore is True if there is AT LEAST one window group queued before the current one
662 else if(iQueue->First()!=this)
666 // fAfter is True if there is MORE THAN one window group queued after the current one
667 TDblQueIter<CWsWindowGroup> iter(*iQueue);
669 if(iter--!=this && iter!=this)
673 TDblQue<CWsWindowGroup>* queue=NULL;
674 // if fBefore and fAfter are True, create a new queue and copy all window groups after the current one into that queue
675 if(fBefore && fAfter)
677 TInt ret=KErrNoMemory;
678 queue=new TDblQue<CWsWindowGroup>(_FOFF(CWsWindowGroup,iChainLink));
681 ret=iChains.Append(queue);
688 // Check that the queue creation and appending worked (we deque all the child groups even if it didn't)
694 // If we've got zero or one window groups after, don't need to queue them
695 if(!fAfter || fBefore)
698 CWsWindowGroup* groupWin;
699 while((groupWin=iter--)!=this)
701 groupWin->iChainLink.Deque();
702 groupWin->iQueue=queue;
704 queue->AddFirst(*groupWin);
707 // if we've got no window groups before, don't need to have a queue for this anymore
719 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
720 if (windowTreeObserver)
722 windowTreeObserver->WindowGroupChainBrokenAfter(*this);
726 else // if this window group isn't queued, we can't clear any children
728 SetReply(KErrArgument);
731 case EWsWinOpSetChildGroup:
733 CWsWindowGroup* childWinGroup = CWsWindowGroup::WindowGroupFromIdentifier(*pData.Int);
734 if(!childWinGroup //(no child to append)
735 || (iQueue && (!iQueue->IsLast(this) || (childWinGroup->iQueue==iQueue))) //(GpWin has a child) || (GpWin and childGpWin in the same queue)
736 || (childWinGroup->iQueue && !childWinGroup->iQueue->IsFirst(childWinGroup)) //(childGpWin has a parent)
737 || (childWinGroup == this)) //(childGpWin == GpWin)
739 SetReply(KErrArgument);
743 // If we have a chain, we're prepending ourselves to the child window group
744 // So we take the childs chain and prepend each of the window groups in our own chain
745 // beginning with the current window group and working backward
747 TDblQueIter<CWsWindowGroup> iter(*iQueue);
749 CWsWindowGroup* groupWin;
750 if(childWinGroup->iQueue)
752 TDblQue<CWsWindowGroup>* oldQueue=iQueue;
753 while((groupWin=iter--)!=NULL)
755 groupWin->iChainLink.Deque();
756 childWinGroup->iQueue->AddFirst(*groupWin);
757 groupWin->iQueue=childWinGroup->iQueue;
759 DeleteQueue(oldQueue);
763 iQueue->AddLast(*childWinGroup);
764 childWinGroup->iQueue=iQueue;
768 // 1. If we don't have a chain, and if the child has a chain, we can simply prepend this wg to the child
770 // 2. If we don't have a chain, and if the child does not have a chain, need to create a chain with the child
771 // as the owning member, and prepend our window group
773 if(childWinGroup->iQueue)
775 childWinGroup->iQueue->AddFirst(*this);
776 iQueue=childWinGroup->iQueue;
780 TDblQue<CWsWindowGroup>* queue=new TDblQue<CWsWindowGroup>(_FOFF(CWsWindowGroup,iChainLink));
781 TInt ret=KErrNoMemory;
784 ret=iChains.Append(queue);
795 queue->AddFirst(*childWinGroup);
796 childWinGroup->iQueue=queue;
797 queue->AddFirst(*this);
803 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
804 if (windowTreeObserver)
806 windowTreeObserver->WindowGroupChained(*this, *childWinGroup);
811 default: // All other window commands disallowed
812 OwnerPanic(EWservPanicOpcode);
816 rootWindow->CheckTree();
820 TPoint CWsWindowGroup::Origin() const
825 TRect CWsWindowGroup::AbsRect() const
827 return (TRect(RootWindow()->Abs().iTl,RootWindow()->Size()));
830 TSize CWsWindowGroup::Size() const
832 return RootWindow()->Size();
835 void CWsWindowGroup::SendState(MWsWindowTreeObserver& aWindowTreeObserver) const
837 aWindowTreeObserver.NodeCreated(*this, ParentNode());
840 void CWsWindowGroup::SendStateWindowGroupChain(MWsWindowTreeObserver& aWindowTreeObserver) const
842 if ( iQueue && (!iQueue->IsEmpty()) && (iQueue->First()==this) )
844 TDblQueIter<CWsWindowGroup> iter(*iQueue);
845 CWsWindowGroup* groupParent;
846 CWsWindowGroup* groupChild;
848 while ( (groupParent=iter++)!=NULL && (groupChild=iter)!=NULL )
850 aWindowTreeObserver.WindowGroupChained(*groupParent, *groupChild);
855 TInt CWsWindowGroup::Identifier() const
860 TPtrC CWsWindowGroup::Name() const
862 return (iGroupName) ? *iGroupName : KNullDesC();
865 TBool CWsWindowGroup::IsFocusable() const
867 return ReceivesFocus();
870 TInt CWsWindowGroup::OrdinalPriority() const
872 return iOrdinalPriorityBase;
875 const MWsClient * CWsWindowGroup::Client() const
877 return static_cast<const MWsClient*>(WsOwner());
881 void CWsWindowGroup::UpdateKeyClickState()
883 CClick::SetKeyClickOveride(iFlags&EGroupFlagDisableKeyClick);
886 void CWsWindowGroup::AreaCovered(TRegion &aRegion)
889 for(CWsClientWindow *win=Child();win;win=win->NextSibling())
890 aRegion.Union(*win->BaseArea());
893 void CWsWindowGroup::SetOrdinalPosition(TInt aPos)
895 if (aPos==(TInt)KOrdinalPositionSwitchToOwningWindow)
896 SwitchToOwningWindow(NULL);
898 SetOrdinalPosition(aPos,NULL);
901 TBool CWsWindowGroup::SetOrdinalPosition(TInt aPos,CWsWindowGroup* aClosingWindow)
905 // Remember if the window group tree is actually changed or not, so that we know whether to
906 // check the render orientation after the re-ordering is done ( see CWsTop::CheckRenderOrientation()
907 // call at end of this method
908 TBool changed = CheckOrdinalPositionChange(aPos);
911 ret=DoSetOrdinalPosition1(aPos,aClosingWindow);
914 TDblQueIter<CWsWindowGroup> iter(*iQueue);
915 CWsWindowGroup* group;
920 while ((group=iter--)!=NULL)
926 TInt lastWinGpPos=NumWindowGroupsOnMyScreen(OrdinalPriority())-after;
930 aPos=Min(aPos,lastWinGpPos);
934 CWsWindowGroup* firstForward=iter--;
935 while (firstForward && firstForward->OrdinalPosition(EFalse)<aPos)
944 iter.Set(*firstForward);
945 MoveChainedWindows(iter,ETrue,aPos,aClosingWindow);
946 iter.Set(*firstForward);
949 MoveChainedWindows(iter,EFalse,--aPos,aClosingWindow);
955 while ((group=iter--)!=this)
957 pos2=group->OrdinalPosition(EFalse);
962 WS_ASSERT_DEBUG(ok, EWsPanicGroupWindowChainError);
966 // If the ordinal positions have changed, check to see if there is a new render orientation
967 // and publish it if so
969 CWsTop::CheckRenderOrientation();
975 void CWsWindowGroup::MoveChainedWindows(TDblQueIter<CWsWindowGroup>& aIter,TBool aForward,TInt aPos,CWsWindowGroup* aClosingWindow)
977 CWsWindowGroup* groupWindow;
978 while ((groupWindow=(aForward ? aIter-- : aIter++))!=NULL)
980 groupWindow->DoSetOrdinalPosition1(aPos,aClosingWindow);
981 (aForward ? ++aPos : --aPos);
985 TBool CWsWindowGroup::DoSetOrdinalPosition1(TInt aPos,CWsWindowGroup* aClosingWindow)
988 if (CheckOrdinalPositionChange(aPos))
990 if (Child()) // A group window with no children can not affect shadows
994 DoSetOrdinalPosition2(aPos,aClosingWindow);
996 else if (aClosingWindow) // do not reset focus if current groupwindow did not change its ordinal position
997 iScreen->ResetFocus(aClosingWindow);
1001 void CWsWindowGroup::DoSetOrdinalPosition2(TInt aPos, CWsWindowGroup *aClosingWindow)
1003 ChangeWindowPosition(aPos,iParent);
1004 ResetFocus(aClosingWindow);
1007 void CWsWindowGroup::LostFocus()
1009 iTextCursor.LostFocus();
1010 iWsOwner->UpdateWindowOrdinalPrioritys();
1011 if (iClientHandle!=0)
1012 QueueEvent(EEventFocusLost);
1013 TWalkWindowTreeFocusChanged wwt(EFalse);
1014 WalkWindowTree(wwt,EWalkChildren);
1015 iWsOwner->SetClientPriority();
1018 void CWsWindowGroup::ReceivedFocus()
1020 iWsOwner->UpdateWindowOrdinalPrioritys();
1021 iTextCursor.ReceivedFocus();
1022 // Used for event queue testing
1023 // Calling MoveToFront sets the queue of the focused window to first place,
1024 // not doing so puts the queues in unusual situation thus stress testing the queue code.
1025 // One such situation is where the focus queue is first but there is a gap before it (iEventPtr>iGlobalEventPtr)"
1027 if ((iEventQueueTest) && (++iSkipCount==5))
1033 WsOwner()->EventQueue()->MoveToFront();
1036 WsOwner()->EventQueue()->MoveToFront();
1039 QueueEvent(EEventFocusGained);
1040 TWalkWindowTreeFocusChanged wwt(ETrue);
1041 WalkWindowTree(wwt,EWalkChildren);
1042 iWsOwner->SetClientPriority();
1043 UpdateKeyClickState();
1046 TInt CWsWindowGroup::NumWindowGroups(TBool aAllPriorities, TInt aPriority)
1050 for(screenNo=0;screenNo<CWsTop::NumberOfScreens();++screenNo)
1052 count+=CWsWindowGroup::NumWindowGroupsOnScreen(CWsTop::Screen(screenNo)->RootWindow()->Child(),aAllPriorities,aPriority);
1057 TInt CWsWindowGroup::NumWindowGroupsOnScreen(const CWsWindowGroup* aGroupWin,TBool aAllPriorities,TInt aPriority)
1062 if (aAllPriorities || aGroupWin->iOrdinalPriority==aPriority)
1064 aGroupWin=aGroupWin->NextSibling();
1069 inline TInt CWsWindowGroup::NumWindowGroupsOnMyScreen(TInt aPriority)
1071 return(CWsWindowGroup::NumWindowGroupsOnScreen(Parent()->Child(),EFalse,aPriority));
1074 void CWsWindowGroup::GetFocusWindowGroupL(TInt aScreenNumber)
1076 CWsWindowGroup *groupWin=(aScreenNumber==KDummyScreenNumber)?CWsTop::FocusWindowGroup():CWsTop::Screen(aScreenNumber)->FocusWindowGroup();
1078 User::Leave(KErrGeneral);
1079 CWsClient::SetReply(groupWin->Identifier());
1082 TInt CWsWindowGroup::GetWindowGroupListL(TInt aScreenNo,TBool aAllPriorities,TInt aPriority,TInt aCount,CArrayFixFlat<TInt>* aList)
1084 TInt count=aList->Count();
1085 CWsWindowGroup* groupWin=CWsTop::Screen(aScreenNo)->RootWindow()->Child();
1086 while(!aAllPriorities && groupWin && groupWin->iOrdinalPriority!=aPriority)
1087 groupWin=groupWin->NextSibling();
1088 while(groupWin && (aAllPriorities || groupWin->iOrdinalPriority==aPriority) && count<aCount)
1090 aList->AppendL(groupWin->Identifier());
1092 groupWin=groupWin->NextSibling();
1097 TInt CWsWindowGroup::SendWindowGroupListL(TInt aScreenNumber, TBool aAllPriorities, TInt aPriority, TInt aCount)
1099 if ((aCount<1) || (aCount>(KArrayMaxGranularity/sizeof(TInt))))
1100 User::Leave(KErrArgument);
1101 CArrayFixFlat<TInt>* list=new(ELeave) CArrayFixFlat<TInt>(aCount);
1102 CleanupStack::PushL(list);
1104 TInt requestedScreen=aScreenNumber;
1105 if(requestedScreen==KDummyScreenNumber)
1107 // get list from current focus screen first
1108 TInt focusScreenNo=CWsTop::CurrentFocusScreen()->ScreenNumber();
1109 count=GetWindowGroupListL(focusScreenNo, aAllPriorities, aPriority, aCount, list);
1112 // now get from the remaining screen
1114 for(screenNo=0;screenNo<CWsTop::NumberOfScreens() && count<aCount;++screenNo)
1116 // skip focus screen
1117 if (screenNo==focusScreenNo)
1119 // count hold total number of window groups collected so far
1120 count=GetWindowGroupListL(screenNo, aAllPriorities, aPriority, aCount, list);
1126 count=GetWindowGroupListL(requestedScreen, aAllPriorities, aPriority, aCount, list);
1128 if (list->Count() > 0)
1129 CWsClient::ReplyBuf(&list->At(0),count*sizeof(TInt));
1130 CleanupStack::PopAndDestroy(list);
1131 return(count); // How many actually returned, may be less than asked for, but not more
1134 void CWsWindowGroup::GetWindowGroupListAndChainL(TInt aScreen,TBool aAllPriorities,TInt aPriority
1135 ,RArray<RWsSession::TWindowGroupChainInfo>& list,TInt& aCountLeft)
1137 CWsWindowGroup *groupWin=CWsTop::Screen(aScreen)->RootWindow()->Child();
1138 while(!aAllPriorities && groupWin && groupWin->iOrdinalPriority!=aPriority)
1139 groupWin=groupWin->NextSibling();
1140 while(groupWin && (aAllPriorities || groupWin->iOrdinalPriority==aPriority) && aCountLeft>0)
1142 RWsSession::TWindowGroupChainInfo windowId;
1143 windowId.iId=groupWin->Identifier();
1144 if(!groupWin->IsChained(windowId.iParentId))
1145 windowId.iParentId=-1; //Unchained window group
1146 list.AppendL(windowId);
1148 groupWin=groupWin->NextSibling();
1152 TInt CWsWindowGroup::SendWindowGroupListAndChainL(TBool aAllPriorities, TInt aPriority, TInt aCount)
1154 if ((aCount<1) || (aCount>(KArrayMaxGranularity/sizeof(RWsSession::TWindowGroupChainInfo))))
1155 User::Leave(KErrArgument);
1156 RArray<RWsSession::TWindowGroupChainInfo> list(aCount);
1157 CleanupClosePushL(list);
1159 TInt focusScreenNo=CWsTop::CurrentFocusScreen()->ScreenNumber();
1160 GetWindowGroupListAndChainL(focusScreenNo,aAllPriorities,aPriority,list,count);
1162 for(screenNo=0;screenNo<CWsTop::NumberOfScreens();++screenNo)
1164 if (screenNo!=focusScreenNo)
1165 GetWindowGroupListAndChainL(screenNo,aAllPriorities,aPriority,list,count);
1168 CWsClient::ReplyBuf(&list[0],aCount*sizeof(RWsSession::TWindowGroupChainInfo));
1169 CleanupStack::PopAndDestroy(&list);
1170 return(aCount-count); // How many actually returned, may be less than asked for, but not more
1173 TBool CWsWindowGroup::SendEventToAllGroups(TBool aAllPriorities,TBool aOnePerClient,const TWsClCmdSendEventToWindowGroup& aData)
1175 TWsEvent event=aData.event;
1176 if (event.Type()==EEventKey && event.Key()->iRepeats!=0)
1177 CKeyboardRepeat::CancelRepeat(NULL); //Otherwise we will trip an invarient
1178 TInt priority=aData.parameter;
1179 TBool sentToAll=ETrue;
1181 for(screenNo=0;screenNo<CWsTop::NumberOfScreens();++screenNo)
1183 CWsWindowGroup *groupWin=CWsTop::Screen(screenNo)->RootWindow()->Child();
1184 if (!aAllPriorities)
1186 while(groupWin && groupWin->iOrdinalPriority!=priority)
1187 groupWin=groupWin->NextSibling();
1189 CWsWindowGroup* firstGroupWin=groupWin;
1190 CWsClient* lastOwner=NULL;
1191 CWsWindowGroup* groupWin2;
1192 while(groupWin && (aAllPriorities || groupWin->iOrdinalPriority==priority))
1196 if (lastOwner==groupWin->iWsOwner)
1198 lastOwner=groupWin->iWsOwner;
1199 for(groupWin2=firstGroupWin;groupWin2!=groupWin;groupWin2=groupWin2->NextSibling())
1201 if (groupWin2->iWsOwner==groupWin->iWsOwner)
1204 if (groupWin2->iWsOwner==groupWin->iWsOwner && groupWin2!=groupWin)
1207 event.SetHandle(groupWin->ClientHandle());
1208 if (!groupWin->EventQueue()->QueueEvent(event))
1211 groupWin=groupWin->NextSibling();
1217 void CWsWindowGroup::ReleasePendedMessagesToAllGroups(CWsClient * aClient)
1220 for (screenNo = 0; screenNo < CWsTop::NumberOfScreens(); ++screenNo)
1222 CWsWindowGroup* groupWin = CWsTop::Screen(screenNo)->RootWindow()->Child();
1225 if (groupWin->WsOwner() == aClient)
1227 groupWin->ReleasePendedMessage();
1229 groupWin = groupWin->NextSibling();
1234 void CWsWindowGroup::ReleasePendedMessage()
1236 if (iMessageArray->Count() > 0 && !(iFlags & EGroupFlagMessageSignalled))
1238 if (!SignalMessageReady())
1240 //The event queue is overflow
1241 // Cannot send a message notification event.
1242 WsOwner()->WgMsgQueueOverflow(); // Set flag for client about having pended message(s)
1247 void CWsWindowGroup::SendMessageToAllGroupsL(CWsClient& aSender,TBool aAllPriorities,const TWsClCmdSendMessageToWindowGroup& aData)
1250 for(screenNo=0;screenNo<CWsTop::NumberOfScreens();++screenNo)
1252 CWsWindowGroup* groupWin=CWsTop::Screen(screenNo)->RootWindow()->Child();
1253 if (!aAllPriorities)
1255 while(groupWin && groupWin->iOrdinalPriority!=aData.identifierOrPriority)
1256 groupWin=groupWin->NextSibling();
1258 while(groupWin && (aAllPriorities || (groupWin->iOrdinalPriority==aData.identifierOrPriority)))
1260 groupWin->QueueMessageL(aData.uid, aData.dataLength, aSender);
1261 groupWin=groupWin->NextSibling();
1266 CWsWindowGroup *CWsWindowGroup::WindowGroupFromIdentifier(TInt aIdentifier)
1268 // apply to all screens
1270 for (screenNo=0; screenNo<CWsTop::NumberOfScreens(); ++screenNo)
1272 CWsWindowGroup* group;
1273 for(group=CWsTop::Screen(screenNo)->RootWindow()->Child(); group; group=group->NextSibling())
1275 if (group->Identifier() == aIdentifier)
1284 CWsWindowGroup *CWsWindowGroup::WindowGroupFromIdentifierL(TInt aIdentifier)
1286 CWsWindowGroup *group=WindowGroupFromIdentifier(aIdentifier);
1288 User::Leave(KErrNotFound);
1292 CWsWindowGroup *CWsWindowGroup::FindWindowGroupL(CWsClient* aClient, TInt aIdentifier,TInt aOffset,const TPtrC *aMatch,const TThreadId *aThreadId)
1294 CWsWindowGroup *group;
1297 group=WindowGroupFromIdentifier(aIdentifier);
1298 if (group) // NULL group will cause KErrNotFound to be returned
1299 group=group->NextSibling();
1303 // get window group for this session
1305 group = aClient->Screen()->RootWindow()->Child();
1308 for(;group;group=group->NextSibling())
1312 if (group->WsOwner()->Client().Id()==*aThreadId)
1317 const TDesC *groupName=&nullDescriptor;
1318 if (group->GroupName())
1319 groupName=group->GroupName();
1320 if (groupName->Length()>=aOffset && groupName->Mid(aOffset).MatchF(*aMatch)>=0)
1325 User::Leave(KErrNotFound);
1329 void CWsWindowGroup::AddPriorityKeyL(TUint aKeycode, TUint aModifierMask, TUint aModifiers)
1331 iPriorityKeys=new(ELeave) TPriorityKey(aKeycode,aModifierMask,aModifiers,iPriorityKeys);
1334 void CWsWindowGroup::RemovePriorityKey(TUint aKeycode, TUint aModifierMask, TUint aModifiers)
1336 for(TPriorityKey **ppk=&iPriorityKeys;*ppk;ppk=&((*ppk)->iNext))
1337 if ((*ppk)->Equals(aKeycode, aModifierMask, aModifiers))
1339 TPriorityKey *next=(*ppk)->iNext;
1346 void CWsWindowGroup::RemoveAllPriorityKeys()
1348 TPriorityKey *pk=iPriorityKeys;
1351 TPriorityKey *next=pk->iNext;
1357 TBool CWsWindowGroup::CheckForPriorityKey(const TKeyEvent &aKeyEvent)
1359 for(TPriorityKey *pk=iPriorityKeys;pk;pk=pk->iNext)
1361 if (pk->KeyMatches(aKeyEvent))
1363 WsOwner()->PriorityKeyPressed(ClientHandle(), aKeyEvent);
1370 void CWsWindowGroup::StatusDump(TDes &aBuf)
1372 _LIT(KWSERVStatusDumpWindowGroupInfo,"CWsWindowGroup[0x%x]RWindowGroup[0x%x,%d],Pri=%d,Id=%d,SizeMode=%d");
1373 aBuf.AppendFormat(KWSERVStatusDumpWindowGroupInfo,this,iClientHandle,LogHandle(),iOrdinalPriority,iIdentifier,iScreenDevice?iScreenDevice->AppMode():0);
1376 TBool CWsWindowGroup::SignalMessageReady()
1379 event.SetType(EEventMessageReady);
1380 event.SetHandle(ClientHandle());
1382 SEventMessageReady& eventMessageReady=*(SEventMessageReady*)event.EventData();
1383 eventMessageReady.iWindowGroupIdentifier=Identifier();
1384 eventMessageReady.iMessageUid=(*iMessageArray)[0].iUid;
1385 eventMessageReady.iMessageParametersSize=iMessageArray->Length(0)-sizeof(TUid);
1386 TBool result = WsOwner()->EventQueue()->QueueEvent(event, EEventPriorityHigh);
1389 iFlags |= EGroupFlagMessageSignalled;
1394 void CWsWindowGroup::QueueMessageL(TUid aUid, TInt aDataLength, CWsClient& aSender)
1396 WS_ASSERT_DEBUG(iFlags&(EGroupFlagMsgQueueActive|EGroupFlagMsgQueueNew) || iMessageArray->Count()>=1,EWsPanicMsgQueueError);
1397 if (!(iFlags&(EGroupFlagMsgQueueActive|EGroupFlagMsgQueueNew)) && iMessageArray->Count()>=KMaxNumberOfMsgsInInactiveQueue)
1399 WS_ASSERT_DEBUG(iMessageArray->Count()<=KMaxNumberOfMsgsInInactiveQueue,EWsPanicMsgQueueError);
1400 iMessageArray->Delete(1,iMessageArray->Count()-1);
1402 TWsMessage* message=NULL;
1403 TRAPD(err,message=&iMessageArray->ExtendL(aDataLength+sizeof(aUid)));
1404 if ((err || (iFlags&EGroupFlagMsgQueueNew)) && iMessageArray->Count()>KMaxNumberOfMsgsInQueue)
1406 iFlags&=~(EGroupFlagMsgQueueActive|EGroupFlagMsgQueueNew);
1407 iMessageArray->Delete(1,iMessageArray->Count()-(err?1:2));
1408 iMessageArray->Compress();
1410 User::LeaveIfError(err);
1414 TPtr8 ptr(&message->iTheRest[0],aDataLength);
1415 TRAP(err,aSender.RemoteReadL(ptr,0));
1418 iMessageArray->Delete(iMessageArray->Count()-1);
1421 if (!(iFlags & EGroupFlagMessageSignalled))
1423 if (!SignalMessageReady())
1425 //The event queue is overflow
1426 // Cannot send a message notification event.
1427 WsOwner()->WgMsgQueueOverflow(); // Set flag for client about having pended message(s)
1433 void CWsWindowGroup::FetchMessageL()
1435 if (!(iFlags & EGroupFlagMessageSignalled))
1437 OwnerPanic(EWservPanicFetchMessage);
1439 CWsClient::ReplyBuf(&((*iMessageArray)[0].iTheRest[0]), (TInt) iMessageArray->Length(0) - sizeof(TUid));
1440 iMessageArray->Delete(0);
1441 iFlags |= EGroupFlagMsgQueueActive;
1442 iFlags &= ~(EGroupFlagMessageSignalled | EGroupFlagMsgQueueNew);
1443 if (iMessageArray->Count() > 0)
1445 if (!SignalMessageReady())
1447 //The event queue is overflow
1448 // Cannot send a message notification event.
1449 WsOwner()->WgMsgQueueOverflow(); // Set flag for client about having pended message(s)
1454 TBool CWsWindowGroup::ScreenDeviceValid() const
1456 return(iScreenDevice?iScreenDevice->ScreenDeviceValidState():(iScreen->ScreenSizeMode()==0));
1459 TBool CWsWindowGroup::CanReceiveFocus() const
1461 return(ReceivesFocus() && iWsOwner->NotClosing() && (ScreenDeviceValid() || iFlags&EGroupFlagHandlesDeviceChange));
1464 void CWsWindowGroup::SetScreenChangeEventStateL(TBool aEnabled)
1466 iFlags&=~EGroupFlagHandlesDeviceChange;
1469 iFlags|=EGroupFlagHandlesDeviceChange;
1470 TWindowServerEvent::AddToScreenDeviceChangeEventListL(*this);
1471 if (iScreen->ScreenSizeMode()!=0)
1472 TWindowServerEvent::SendScreenDeviceChangedEvent(this);
1475 TWindowServerEvent::RemoveFromScreenDeviceChangeEventList(*this);
1476 iScreen->ResetFocus(NULL);
1479 void CWsWindowGroup::SetScreenDeviceValidState(const DWsScreenDevice *aDevice)
1481 if (iScreenDevice==aDevice)
1483 TBool state=ScreenDeviceValid();
1484 for(CWsTopClientWindow *win=Child();win;win=win->NextSiblingTop())
1486 win->SetScreenDeviceValidState(state);
1491 void CWsWindowGroup::SetScreenDeviceValidStates(const DWsScreenDevice *aDevice)
1493 for(CWsWindowGroup *groupWin=aDevice->RootWindow()->Child();groupWin;groupWin=groupWin->NextSibling())
1494 groupWin->SetScreenDeviceValidState(aDevice);
1497 void CWsWindowGroup::SetScreenDeviceValidStates(CScreen* aScreen)
1499 CWsRootWindow* rootWindow = aScreen->RootWindow();
1501 CWsWindowGroup* groupWin;
1502 CWsTopClientWindow* win;
1503 for(groupWin=rootWindow->Child();groupWin;groupWin=groupWin->NextSibling())
1505 TBool state=groupWin->ScreenDeviceValid();
1506 for(win=groupWin->Child();win;win=win->NextSiblingTop())
1508 win->SetScreenDeviceValidStateFlag(state);
1509 win->ResetHiddenFlagsInParentAndChildren();
1514 void CWsWindowGroup::SetScreenDevice(DWsScreenDevice *aDevice)
1516 iScreenDevice=aDevice;
1519 void CWsWindowGroup::NewOrientation(TInt aMode,CFbsBitGc::TGraphicsOrientation aRotation, CWsRootWindow* aRootWindow)
1521 for(CWsWindowGroup *groupWin=aRootWindow->Child();groupWin;groupWin=groupWin->NextSibling())
1523 DWsScreenDevice *device=groupWin->Device();
1525 device->NewOrientation(aMode,aRotation);
1529 void CWsWindowGroup::ResetFocus(CWsWindowGroup *aClosingWindow)
1533 iScreen->ResetFocus(aClosingWindow);
1537 TBool CWsWindowGroup::IsChained(TInt& aParentId)
1541 if (iQueue->First()==this)
1544 aParentId=BeforeInChain()->Identifier();
1548 inline CWsWindowGroup* CWsWindowGroup::BeforeInChain()
1549 { //You should only call this function if you know the window has a parent
1550 return reinterpret_cast<CWsWindowGroup*>(PtrSub(iChainLink.iPrev,_FOFF(CWsWindowGroup,iChainLink)));
1553 TBool CWsWindowGroup::CheckCapability(TInt& aOrdinalPriority)
1555 if(aOrdinalPriority>=KPasswordWindowGroupPriority)
1557 if(!KSecurityPolicy_SwEvent().CheckPolicy(WsOwner()->Client(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed")))
1559 aOrdinalPriority=KPasswordWindowGroupPriority-1;
1566 TBool CWsWindowGroup::HasVisibleTranslucentChild()
1568 CWsWindowBase * child = iChild;
1571 if (child->WinType() == EWinTypeClient)
1573 CWsClientWindow * cliwin = static_cast<CWsClientWindow *>(child);
1574 if (cliwin->IsTranslucent() && cliwin->IsVisible())
1577 else if (static_cast<CWsWindowGroup*>(child)->HasVisibleTranslucentChild())
1581 child = child->NextSibling();
1587 TInt CWsWindowGroup::NumClientWindowGroups()
1589 CWsObjectIx& objix=*WsOwner()->ObjectIndex();
1590 const TWsObject* ptr=objix.FirstObject();
1591 const TWsObject* end=ptr+objix.Length();
1593 while(++ptr<end) //First one should always have a NULL object
1595 const CWsObject* obj=ptr->iObject;
1596 if (obj && obj->Type()==WS_HANDLE_GROUP_WINDOW)
1602 void CWsWindowGroup::SetEventQueueTestState(TBool aEventQueState)
1604 iEventQueueTest = aEventQueState;