os/graphics/windowing/windowserver/nga/SERVER/openwfc/GROUPWIN.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Group window sub-class of CWsWindow
    15 // 
    16 //
    17 
    18 #include <e32std.h>
    19 #include "W32STD.H"
    20 #include "W32CLICK.H"
    21 #include "server.h"
    22 #include "rootwin.h"
    23 #include "windowgroup.h"
    24 #include "walkwindowtree.h"
    25 #include "wstop.h"
    26 #include "EVENT.H"
    27 #include "KEYCLICK.H"
    28 #include "PRIKEY.H"
    29 #include "panics.h"
    30 #include "windowelementset.h"
    31 #include "pointer.h"
    32 
    33 GLREF_D TPtr nullDescriptor;
    34 GLREF_D CDebugLogBase* wsDebugLog;
    35 
    36 #if defined(_DEBUG)
    37 TInt CWsWindowGroup::iSkipCount=0;
    38 #endif
    39 
    40 TBool CWsWindowGroup::iEventQueueTest=EFalse;			// For stress testing the EventQueue code
    41 
    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;
    48 
    49 CWsWindowGroup* CWsWindowGroup::NewL(CWsClient* aOwner, CScreen* aScreen, 
    50 	const TWsClCmdCreateWindowGroup& aCmd)
    51 	{
    52 	CWsWindowGroup* self = new(ELeave) CWsWindowGroup(aOwner, aScreen);
    53 	CleanupStack::PushL(self);
    54 	self->ConstructL(aCmd);
    55 	CleanupStack::Pop(self);
    56 	return self;
    57 	}
    58 
    59 CWsWindowGroup::CWsWindowGroup(CWsClient* aOwner, CScreen* aScreen) : CWsWindowBase(aOwner,WS_HANDLE_GROUP_WINDOW,aScreen)
    60 	{
    61 	__DECLARE_NAME(_S("CWsWindowGroup"));
    62 	iWinType=EWinTypeGroup;
    63 	}
    64 
    65 void CWsWindowGroup::PurgeCapturedKeys()
    66 	{
    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
    71 		{
    72 		const CWsObject* obj=ptr->iObject;
    73 		if (obj 
    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)))
    77 			{
    78 			objix.Remove(ptr);
    79 			delete obj;
    80 			}
    81 		}
    82 	objix.Tidy();
    83 	CKeyboardRepeat::CancelRepeat(this);
    84 	}
    85 
    86 void CWsWindowGroup::SwitchToOwningWindow(CWsWindowGroup *aClosingWindow)
    87 	{
    88 	if (this==CWsTop::FocusWindowGroup())
    89 		{
    90 		CWsWindowGroup *winGroup=NULL;
    91 //
    92 // First try for an 'owning' window
    93 //
    94 		if (iOwningWindowGroup)
    95 			{
    96 			for(winGroup=RootWindow()->Child();winGroup;winGroup=winGroup->NextSibling())
    97 				if (winGroup->Identifier()==iOwningWindowGroup && winGroup->iOrdinalPriority==iOrdinalPriority)
    98 					goto gotIt;
    99 			}
   100 //
   101 // If that failed look for the frontmost window belonging to the owner of dying window
   102 //
   103 		for(winGroup=RootWindow()->Child();winGroup;winGroup=winGroup->NextSibling())
   104 			if (winGroup!=this && winGroup->WsOwner()==WsOwner() && winGroup->iOrdinalPriority==iOrdinalPriority)
   105 				goto gotIt;
   106 //
   107 // Next try for the nominated default owning window group
   108 //
   109 		winGroup=iScreen->DefaultOwningWindowGroup();
   110 		if (winGroup && winGroup->iOrdinalPriority==iOrdinalPriority)
   111 			{
   112 gotIt:		winGroup->SetOrdinalPosition(0,this);
   113 			return;
   114 			}
   115 		}
   116 	ResetFocus(aClosingWindow);
   117 	}
   118 
   119 CWsWindowGroup::~CWsWindowGroup()
   120 	{
   121 	MWsWindowTreeObserver* windowTreeObserver = NULL;
   122 	if (Screen()&& iBaseWinFlags&EBaseWinNodeCreated)
   123 		{
   124 		windowTreeObserver = Screen()->WindowTreeObserver();
   125 		if (windowTreeObserver)
   126 			{
   127 			if( iQueue && (iQueue->Last()!=this) )
   128 				{
   129 				//This node is part of chain, send notification unless this is the last node in the chain
   130 				windowTreeObserver->WindowGroupChainBrokenAfter(*this);
   131 				}
   132 			windowTreeObserver->NodeReleased(*this);
   133 			iBaseWinFlags &= ~EBaseWinNodeCreated;
   134 			}
   135 		}
   136 	DisconnectFloatingSprites();
   137 	
   138 	if (wsDebugLog)
   139 		{
   140 		TLogMessageText buf;
   141 		_LIT(KWSERVDebugLogGroupWindowId,"Destroying: RWindowGroup[0x%x,%d],Id=%d");
   142 		buf.Format(KWSERVDebugLogGroupWindowId,iClientHandle,LogHandle(),iIdentifier);
   143 		wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, buf);
   144 		}
   145 	if (CClick::IsHandler())
   146 		CClick::OtherEvent(EEventGroupWindowClose,reinterpret_cast<TAny*>(iIdentifier));
   147 	if (iQueue)
   148 		{
   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;
   153 			iter.SetToLast();
   154 			while ((groupWin=iter--)!=this)
   155 				{
   156 				WS_ASSERT_DEBUG(groupWin!=NULL && groupWin->iQueue==iQueue,EWsPanicGroupWindowChainError);
   157 				if ( windowTreeObserver && (iQueue->Last()!=groupWin) )
   158 					{
   159 					//Send notification unless groupWin is the last node in the chain
   160 					windowTreeObserver->WindowGroupChainBrokenAfter(*groupWin);
   161 					}
   162 				groupWin->iChainLink.Deque();
   163 				// coverity[extend_simple_error]
   164 				groupWin->iQueue=NULL;
   165 				}
   166 			}
   167 		WS_ASSERT_DEBUG(iQueue->Last()==this,EWsPanicGroupWindowChainError);
   168 		if(windowTreeObserver && !iQueue->IsEmpty())
   169 			{
   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;
   175 			if(parent)
   176 				{
   177 				windowTreeObserver->WindowGroupChainBrokenAfter(*parent);
   178 				}
   179 			}
   180 		TDblQueLinkBase* parentLink=iChainLink.iPrev;
   181 		iChainLink.Deque();
   182 		if (parentLink->iNext==parentLink->iPrev)		//Check to see chain no longer required
   183 			{
   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]
   190 				parent->iQueue=NULL;
   191 				}
   192 			DeleteQueue(iQueue);
   193 			}
   194 		}
   195 	RemoveAllPriorityKeys();
   196 	PurgeCapturedKeys();
   197 	iTextCursor.Close();
   198 	SetPointerCursor(NULL);
   199 	for(CWsTopClientWindow *win=Child();win;win=win->NextSiblingTop())
   200 		win->SetInactive();
   201 	if (iScreen)
   202 		{
   203 		iScreen->RemoveFromDefaultOwningList(this);
   204 		}
   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);
   212 	delete iGroupName;
   213 	delete iMessageArray;
   214 	}
   215 
   216 void CWsWindowGroup::DisconnectFloatingSprites()
   217 	{
   218 	CWsSpriteBase * current = iSpriteList;
   219 	while (current)
   220 		{
   221 		CWsSpriteBase * next = current->Next();
   222 		current->Deactivate();
   223 		current->DisconnectGroupWin();
   224 		current = next;
   225 		}
   226 	}
   227 
   228 void CWsWindowGroup::DeleteQueue(TDblQue<CWsWindowGroup>* aQueue)
   229 	{
   230 	iChains.Remove(iChains.Find(aQueue));
   231 	delete aQueue;
   232 	if (iChains.Count()==0)
   233 		{
   234 		iChains.Compress();
   235 		}
   236 	}
   237 
   238 void CWsWindowGroup::AdvanceIdentifierCount()
   239 	{
   240 	if (++iIdentifierCount>EMaxIdentifierCount)
   241 		iIdentifierCount=1;		// so limit it to low value
   242 	}
   243 
   244 void CWsWindowGroup::ConstructL(const TWsClCmdCreateWindowGroup &aCmd)
   245 	{
   246 #if defined(_DEBUG)
   247 	if (IsClientHandleInUse(aCmd.clientHandle))
   248 		{
   249 		OwnerPanic(EWservPanicDuplicateHandle);
   250 		}
   251 #endif
   252 	NewObjL();
   253 	iFlags=EGroupFlagAutoForeground|EGroupFlagMsgQueueNew;
   254 	if (aCmd.focus)
   255 		{
   256 		iFlags|=EGroupFlagReceivesFocus;
   257 		}
   258 	iTextCursor.ConstructL(this);
   259 	iClientHandle=aCmd.clientHandle;
   260 
   261 	if(aCmd.screenDeviceHandle <= 0)
   262 		{
   263 		//Use primary screen. Client should make sure PrimaryScreenDevice is correct set up immediately after establishing session.
   264 		iScreenDevice=iWsOwner->PrimaryScreenDevice();
   265 		}
   266 	else
   267 		{
   268 		//Use the specified screen
   269 		iScreenDevice=STATIC_CAST(DWsScreenDevice*,iWsOwner->HandleToObj(aCmd.screenDeviceHandle,WS_HANDLE_SCREEN_DEVICE));
   270 		}
   271 
   272 	iScreen = (iScreenDevice) ? iScreenDevice->Screen() : CWsTop::Screen(); //if no screen device use screen 0
   273 
   274 	CWsWindowGroup* parent=NULL;
   275 	if (aCmd.parentId>0)
   276 		{
   277 		parent=CWsWindowGroup::WindowGroupFromIdentifier(aCmd.parentId);
   278 		if (!parent)
   279 			{
   280 			OwnerPanic(EWservPanicWindow);
   281 			}
   282 		
   283  		if(parent->Screen() != iScreen)
   284  			{
   285  			OwnerPanic(EWservPanicWrongScreen);
   286  			}
   287 		
   288 		if (parent->iOrdinalPriorityAdjust>0)
   289 			{
   290 			WS_ASSERT_DEBUG(parent->iQueue==NULL,EWsPanicGroupWindowChainError);
   291 			parent->iOrdinalPriorityAdjust=0;
   292 			parent->UpdateOrdinalPriority(ETrue);
   293 			}
   294 		iOrdinalPriorityBase=parent->iOrdinalPriorityBase;
   295 		iOrdinalPriority=iOrdinalPriorityBase;
   296 		}
   297 
   298 	do
   299 		{
   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;
   303 
   304 	CWsWindowBase::ConstructL(RootWindow());
   305 	iScreen->ResetFocus(NULL);
   306 	
   307 	if (parent)
   308 		{
   309 		TDblQue<CWsWindowGroup>* queue=parent->iQueue;
   310 		if (queue && queue->Last()!=parent)
   311 			User::Leave(KErrInUse);
   312 		if (parent->iWsOwner!=iWsOwner)
   313 			{
   314 			_LIT_SECURITY_POLICY_S0(securityPolicy,parent->iChildSID);
   315 			if (!securityPolicy().CheckPolicy(iWsOwner->ClientMessage()))
   316 				User::Leave(KErrPermissionDenied);
   317 			}
   318 		if (!queue)
   319 			{
   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;			
   326 			}
   327 		iQueue=queue;		//Shouldn't set the queue until after it can leave
   328 		iChainLink.Enque(&parent->iChainLink);
   329 		
   330 		MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();						
   331 		if (windowTreeObserver)
   332 			{
   333 			windowTreeObserver->WindowGroupChained(*parent, *this);
   334 			}			
   335 		}
   336 	iMessageArray=new(ELeave) CArrayVarSeg<TWsMessage>(1);
   337 	if (CClick::IsHandler())
   338 		{
   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,&params);
   344 		}
   345 	if (wsDebugLog)
   346 		{
   347 		TLogMessageText buf;
   348 		_LIT(KWSERVDebugLogGroupWindowId,"Creating: RWindowGroup[0x%x,%d],Id=%d");
   349 		buf.Format(KWSERVDebugLogGroupWindowId,iClientHandle,LogHandle(),iIdentifier);
   350 		wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, buf);
   351 		}
   352 	}
   353 
   354 void CWsWindowGroup::UpdateOrdinalPriority(TBool aDoAdjust)
   355 	{
   356 	TInt newPri;
   357 	newPri=iOrdinalPriorityBase;
   358 	if (iWsOwner==CWsTop::FocusWindowGroupOwner())
   359 		newPri+=iOrdinalPriorityAdjust;
   360 	CheckCapability(newPri);
   361 	if (newPri!=iOrdinalPriority)
   362 		{
   363 		iOrdinalPriority=newPri;
   364 		if (aDoAdjust)
   365 			SetOrdinalPosition(0);
   366 		}
   367 	}
   368 
   369 void CWsWindowGroup::SetOrdinalPriority(TInt aPos,TInt aPriority)
   370 	{
   371 	if (!iQueue)
   372 		{
   373 		iOrdinalPriorityBase=aPriority;
   374 		UpdateOrdinalPriority(EFalse);
   375 		}
   376 	else
   377 		{
   378 		TDblQueIter<CWsWindowGroup> iter(*iQueue);
   379 		CWsWindowGroup* group;
   380 		while ((group=iter++)!=NULL)
   381 			{
   382 			group->iOrdinalPriorityBase=aPriority;
   383 			group->iOrdinalPriority=aPriority;
   384 			}
   385 		}
   386 	SetOrdinalPosition(aPos);
   387 	}
   388 
   389 void CWsWindowGroup::CommandL(TInt aOpcode, const TAny *aCmdData)
   390 	{
   391 #ifdef _DEBUG
   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();
   396 
   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.
   400 	switch (aOpcode)
   401 		{
   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:
   420 			{
   421 			if (ScreenDeviceDeleted())
   422 				OwnerPanic(EWservPanicGroupWinScreenDeviceDeleted);
   423 			break;				
   424 			};
   425 		}
   426 #endif
   427 
   428 	TWsWinCmdUnion pData;
   429 	pData.any=aCmdData;
   430 	if (CWsWindowBase::CommandL(aOpcode,pData)==EFalse)
   431 		{
   432 		switch(aOpcode)
   433 			{
   434 			case EWsWinOpAllowChildWindowGroup:
   435 				iChildSID=*pData.UInt;
   436 				break;
   437 			case EWsWinOpEnableScreenChangeEvents:
   438 				SetScreenChangeEventStateL(ETrue);
   439 				break;
   440 			case EWsWinOpDisableScreenChangeEvents:
   441 				SetScreenChangeEventStateL(EFalse);
   442 				break;
   443 			case EWsWinOpReceiveFocus:
   444 				if (*pData.Bool!=(iFlags&EGroupFlagReceivesFocus))
   445 					{
   446 					iFlags&=~EGroupFlagReceivesFocus;
   447 					if (*pData.Bool)
   448 						iFlags|=EGroupFlagReceivesFocus;
   449 					iScreen->ResetFocus(NULL);
   450 					}
   451 				break;
   452 			case EWsWinOpAutoForeground:
   453 				iFlags&=~EGroupFlagAutoForeground;
   454 				if (*pData.Bool)
   455 					iFlags|=EGroupFlagAutoForeground;
   456 				break;
   457 			case EWsWinOpSetOrdinalPositionPri:
   458 			case EWsWinOpSetOrdinalPositionErr:
   459 				{
   460 				TInt priority=pData.OrdinalPos->ordinalPriority;
   461 				TBool hascap = CheckCapability(priority);
   462 				SetOrdinalPriority(pData.OrdinalPos->pos, priority);
   463 				if (aOpcode == EWsWinOpSetOrdinalPositionErr)
   464 					{
   465 					SetReply(hascap?KErrNone:KErrPermissionDenied);
   466 					}
   467 				}
   468 				break;
   469 			case EWsWinOpSetOrdinalPriorityAdjust:
   470 				if (!iQueue)
   471 					{
   472 					iOrdinalPriorityAdjust=*pData.Int;
   473 					UpdateOrdinalPriority(ETrue);
   474 					}
   475 				break;
   476 			case EWsWinOpCaptureKey:
   477 				{
   478 				if(!KSecurityPolicy_SwEvent().CheckPolicy(iWsOwner->ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWindowGroup::CaptureKey API")))
   479 					{
   480 					User::Leave(KErrPermissionDenied);
   481 					}
   482 				CWsCaptureKey *cKey=new(ELeave) CWsCaptureKey(this);
   483 				CleanupStack::PushL(cKey);
   484 				cKey->ConstructL(*pData.CaptureKey);
   485 				CleanupStack::Pop();
   486 				}
   487 				break;
   488 			case EWsWinOpCaptureKeyUpsAndDowns:
   489 				{
   490 				if(!KSecurityPolicy_SwEvent().CheckPolicy(iWsOwner->ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWindowGroup::CaptureKeyUpsAndDowns API")))
   491 					{
   492 					User::Leave(KErrPermissionDenied);
   493 					}
   494 				CWsCaptureKeyUpsAndDowns *cKey=new(ELeave) CWsCaptureKeyUpsAndDowns(this);
   495 				CleanupStack::PushL(cKey);
   496 				cKey->ConstructL(*pData.CaptureKey);
   497 				CleanupStack::Pop();
   498 				}
   499 				break;
   500 			case EWsWinOpCaptureLongKey:
   501 				{
   502 				if(!KSecurityPolicy_SwEvent().CheckPolicy(iWsOwner->ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWindowGroup::CaptureLongKey API")))
   503 					{
   504 					User::Leave(KErrPermissionDenied);
   505 					}
   506 				CWsCaptureLongKey *cKey=new(ELeave) CWsCaptureLongKey(this);
   507 				CleanupStack::PushL(cKey);
   508 				cKey->ConstructL(*pData.CaptureLongKey);
   509 				CleanupStack::Pop();
   510 				}
   511 				break;
   512 			case EWsWinOpCancelCaptureKey:
   513 				if (*pData.UInt!=0)	// Ignore null handle
   514 					{
   515 					CWsObject *destroyObj = iWsOwner->HandleToObj(*pData.UInt, WS_HANDLE_CAPTURE_KEY);
   516 					if (destroyObj)
   517 						{
   518 						// Cancel any repeat that is underway for this capture
   519 						CKeyboardRepeat::CancelRepeat(destroyObj, EFalse);
   520 						delete destroyObj;
   521 						}
   522 					else
   523 						{
   524 #ifdef _DEBUG
   525 						// Attempt to cancel key capture with an incorrect handle
   526 						OwnerPanic(EWservPanicDestroy);  
   527 #endif // _DEBUG
   528 						}					
   529 					}
   530 				break;
   531 			case EWsWinOpCancelCaptureKeyUpsAndDowns:
   532 				if (*pData.UInt!=0)	// Ignore null handle
   533 					{
   534 					CWsObject *destroyObj = iWsOwner->HandleToObj(*pData.UInt, WS_HANDLE_CAPTURE_KEY_UPDOWNS);
   535 					if (destroyObj)
   536 						{
   537 						delete destroyObj;
   538 						}
   539 					else
   540 						{
   541 #ifdef _DEBUG						
   542 						// Attempt to cancel ups and downs key capture with an incorrect handle
   543 						OwnerPanic(EWservPanicDestroy);
   544 #endif // _DEBUG
   545 						}					
   546 					}
   547 				break;
   548 			case EWsWinOpCancelCaptureLongKey:
   549 				if (*pData.UInt!=0)	// Ignore null handle
   550 					{
   551 					CWsObject *destroyObj = iWsOwner->HandleToObj(*pData.UInt, WS_HANDLE_CAPTURE_LONG_KEY);
   552 					if (destroyObj)
   553 						{
   554 						// Cancel any repeat that is underway for this capture
   555 						CKeyboardRepeat::CancelRepeat(destroyObj, ETrue);
   556 						delete destroyObj;
   557 						} 
   558 					else
   559 						{
   560 #ifdef _DEBUG						
   561 						// Attempt to cancel long key capture with an incorrect handle
   562 						OwnerPanic(EWservPanicDestroy);
   563 #endif // _DEBUG
   564 						}					
   565 					}
   566 				break;
   567 			case EWsWinOpAddPriorityKey:
   568 				AddPriorityKeyL(pData.PriorityKey->keycode, pData.PriorityKey->modifierMask, pData.PriorityKey->modifiers);
   569 				break;
   570 			case EWsWinOpRemovePriorityKey:
   571 				RemovePriorityKey(pData.PriorityKey->keycode, pData.PriorityKey->modifierMask, pData.PriorityKey->modifiers);
   572 				break;
   573 			case EWsWinOpSetTextCursor:
   574 				iTextCursor.SetL(*pData.SetTextCursor, EFalse);
   575 				break;
   576 			case EWsWinOpSetTextCursorClipped:
   577 				iTextCursor.SetL(*pData.SetTextCursor, ETrue);
   578 				break;
   579 			case EWsWinOpCancelTextCursor:
   580 				iTextCursor.Cancel();
   581 				break;
   582 			case EWsWinOpSetOwningWindowGroup:
   583 				iOwningWindowGroup=*pData.Int;
   584 				break;
   585 			case EWsWinOpDefaultOwningWindow:
   586 				{
   587 				if(KSecurityPolicy_WriteDeviceData().CheckPolicy(iWsOwner->ClientMessage(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed for RWindowGroup::DefaultOwningWindow API")))
   588 					{
   589 					iScreen->SetDefaultOwningWindow(this);
   590 					}
   591 				}
   592 				break;
   593 			case EWsWinOpName:
   594 				iWsOwner->ReplyGroupName(iGroupName,*pData.Int);
   595 				break;
   596 			case EWsWinOpSetName:
   597 				{
   598 				HBufC *newName=NULL;
   599 				const TInt size=*pData.Int;
   600 				if (size>0)
   601 					{
   602 					newName=HBufC::NewLC(size);
   603 					TPtr ptr(newName->Des());
   604 					iWsOwner->RemoteReadL(ptr,0);
   605 					CleanupStack::Pop(newName);
   606 					}
   607 				//Window Group Name is unchanged
   608 				if (iGroupName && newName && *iGroupName == *newName)
   609 					{
   610 					delete newName;
   611 					} 
   612 				else  //Window Group Name is changed
   613 					{  
   614 					delete iGroupName;
   615 					iGroupName=newName;
   616 					TWindowServerEvent::SendGroupChangedEvents();
   617 					if (iScreen && iScreen->ChangeTracking())
   618 					    {
   619 				        MWsWindowTreeObserver* const windowTreeObserver = iScreen->WindowTreeObserver();
   620 				        if (windowTreeObserver)
   621 				            windowTreeObserver->AttributeChanged(*this, MWsWindowTreeObserver::EWindowGroupName);
   622 					    }
   623 					}
   624 				}
   625 				break;
   626 			case EWsWinOpIdentifier:
   627 				SetReply(Identifier());
   628 				break;
   629 			case EWsWinOpDisableKeyClick:
   630 				if (*pData.Bool)
   631 					iFlags|=EGroupFlagDisableKeyClick;
   632 				else
   633 					iFlags&=~EGroupFlagDisableKeyClick;
   634 				if (this==CWsTop::FocusWindowGroup())
   635 					UpdateKeyClickState();
   636 				break;
   637 			case EWsWinOpSendAdvancedPointerEvent:
   638 			case EWsWinOpSendPointerEvent:
   639 				{
   640 				TRawEvent eventCopy = *pData.RawEvent;
   641 				if (TWsPointer::PreProcessClientEvent(eventCopy, aOpcode == EWsWinOpSendAdvancedPointerEvent))
   642 					{
   643 					if (!TWindowServerEvent::MousePress(eventCopy,this))
   644 						{
   645 						OwnerPanic(EWservPanicEventType);
   646 						}
   647 					}
   648 				}
   649 				break;
   650 			case EWsWinOpClearChildGroup:
   651 				if(iQueue)
   652 					{
   653 					TBool fBefore=EFalse; 
   654 					TBool fAfter=EFalse; 
   655 					// If there is nothing to clear, return KErrArgument
   656 					if(iQueue->Last()==this)
   657 						{
   658 						SetReply(KErrArgument);
   659 						break;
   660 						}
   661 					// fBefore is True if there is AT LEAST one window group queued before the current one
   662 					else if(iQueue->First()!=this)
   663 						{
   664 						fBefore=ETrue;
   665 						}
   666 					// fAfter is True if there is MORE THAN one window group queued after the current one
   667 					TDblQueIter<CWsWindowGroup> iter(*iQueue);
   668 					iter.SetToLast();
   669 					if(iter--!=this && iter!=this)
   670 						{
   671 						fAfter=ETrue;
   672 						}
   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)
   676 						{
   677 						TInt ret=KErrNoMemory;
   678 						queue=new TDblQue<CWsWindowGroup>(_FOFF(CWsWindowGroup,iChainLink));
   679 						if(queue)
   680 							{
   681 							ret=iChains.Append(queue);	
   682 							if(ret!=KErrNone)
   683 								{
   684 								delete queue;
   685 								queue=NULL; 
   686 								}
   687 							}
   688 						// Check that the queue creation and appending worked (we deque all the child groups even if it didn't) 
   689 						if(ret!=KErrNone)
   690 							{
   691 							SetReply(ret);
   692 							}
   693 						}
   694 					// If we've got zero or one window groups after, don't need to queue them
   695 					if(!fAfter || fBefore)
   696 						{
   697 						iter.SetToLast();
   698 						CWsWindowGroup* groupWin;
   699 						while((groupWin=iter--)!=this)
   700 							{
   701 							groupWin->iChainLink.Deque();
   702 							groupWin->iQueue=queue;
   703 							if(queue)
   704 								queue->AddFirst(*groupWin);
   705 							}
   706 						}
   707 					// if we've got no window groups before, don't need to have a queue for this anymore
   708 					if(!fBefore)
   709 						{
   710 						iChainLink.Deque();
   711 						if (!fAfter)
   712 							{
   713 							DeleteQueue(iQueue);
   714 							}
   715 						iQueue=NULL;						
   716 						}
   717 					if (Screen())
   718 						{
   719 					MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
   720 					if (windowTreeObserver)
   721 						{
   722 						windowTreeObserver->WindowGroupChainBrokenAfter(*this);
   723 							}
   724 						}					
   725 					}
   726 				else // if this window group isn't queued, we can't clear any children
   727 					{
   728 					SetReply(KErrArgument);
   729 					}
   730 				break;
   731 			case EWsWinOpSetChildGroup:
   732 				{
   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)
   738 					{
   739 					SetReply(KErrArgument);
   740 					break;
   741 					}
   742 				if(iQueue)
   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
   746 					{
   747 					TDblQueIter<CWsWindowGroup> iter(*iQueue);
   748 					iter.SetToLast();
   749 					CWsWindowGroup* groupWin;
   750 					if(childWinGroup->iQueue)
   751 						{
   752 						TDblQue<CWsWindowGroup>* oldQueue=iQueue;
   753 						while((groupWin=iter--)!=NULL)
   754 							{
   755 							groupWin->iChainLink.Deque();
   756 							childWinGroup->iQueue->AddFirst(*groupWin);
   757 							groupWin->iQueue=childWinGroup->iQueue;
   758 							}
   759 						DeleteQueue(oldQueue);
   760 						}
   761 					else
   762 						{
   763 						iQueue->AddLast(*childWinGroup);
   764 						childWinGroup->iQueue=iQueue;							
   765 						}
   766 					}
   767 				else
   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
   769 				// wg chain
   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
   772 					{
   773 					if(childWinGroup->iQueue)
   774 						{
   775 						childWinGroup->iQueue->AddFirst(*this);
   776 						iQueue=childWinGroup->iQueue;
   777 						}
   778 					else
   779 						{
   780 						TDblQue<CWsWindowGroup>* queue=new TDblQue<CWsWindowGroup>(_FOFF(CWsWindowGroup,iChainLink));
   781 						TInt ret=KErrNoMemory;
   782 						if (queue)
   783 							{
   784 							ret=iChains.Append(queue);
   785 							if(ret!=KErrNone)
   786 								{
   787 								delete queue;
   788 								}
   789 							}
   790 						if(ret!=KErrNone)
   791 							{
   792 							SetReply(ret);
   793 							break;
   794 							}
   795 						queue->AddFirst(*childWinGroup);
   796 						childWinGroup->iQueue=queue;
   797 						queue->AddFirst(*this);
   798 						iQueue=queue;						
   799 						}						
   800 					}
   801 				if (Screen())
   802 					{
   803 				MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();						
   804 				if (windowTreeObserver)
   805 					{
   806 					windowTreeObserver->WindowGroupChained(*this, *childWinGroup);
   807 						}
   808 					}
   809 				}
   810 				break;
   811 			default:						// All other window commands disallowed
   812 				OwnerPanic(EWservPanicOpcode);
   813 			}
   814 		}
   815 #if defined(_DEBUG)
   816 	rootWindow->CheckTree();
   817 #endif
   818 	}
   819 	
   820 TPoint CWsWindowGroup::Origin() const
   821 	{
   822 	return TPoint(0,0);
   823 	}
   824 	
   825 TRect CWsWindowGroup::AbsRect() const
   826 	{
   827 	return (TRect(RootWindow()->Abs().iTl,RootWindow()->Size()));
   828 	}
   829 
   830 TSize CWsWindowGroup::Size() const
   831 	{
   832 	return RootWindow()->Size();
   833 	}
   834 
   835 void CWsWindowGroup::SendState(MWsWindowTreeObserver& aWindowTreeObserver) const
   836 	{	
   837 	aWindowTreeObserver.NodeCreated(*this, ParentNode());
   838 	}
   839 
   840 void CWsWindowGroup::SendStateWindowGroupChain(MWsWindowTreeObserver& aWindowTreeObserver) const
   841 	{
   842 	if ( iQueue && (!iQueue->IsEmpty()) && (iQueue->First()==this) )
   843 		{
   844 		TDblQueIter<CWsWindowGroup> iter(*iQueue);
   845 		CWsWindowGroup* groupParent;
   846 		CWsWindowGroup* groupChild;
   847 		
   848 		while ( (groupParent=iter++)!=NULL && (groupChild=iter)!=NULL )
   849 			{
   850 			aWindowTreeObserver.WindowGroupChained(*groupParent, *groupChild);
   851 			}
   852 		}
   853 	}
   854 
   855 TInt CWsWindowGroup::Identifier() const
   856 	{
   857 	return iIdentifier;
   858 	}
   859 
   860 TPtrC CWsWindowGroup::Name() const
   861 	{
   862 	return (iGroupName) ? *iGroupName : KNullDesC();
   863 	}
   864 
   865 TBool CWsWindowGroup::IsFocusable() const
   866 	{
   867 	return ReceivesFocus();
   868 	}
   869 
   870 TInt CWsWindowGroup::OrdinalPriority() const
   871 	{
   872 	return iOrdinalPriorityBase;
   873 	}
   874 
   875 const MWsClient * CWsWindowGroup::Client() const
   876     {
   877     return static_cast<const MWsClient*>(WsOwner());
   878     }
   879 
   880 
   881 void CWsWindowGroup::UpdateKeyClickState()
   882 	{
   883 	CClick::SetKeyClickOveride(iFlags&EGroupFlagDisableKeyClick);
   884 	}
   885 
   886 void CWsWindowGroup::AreaCovered(TRegion &aRegion)
   887 	{
   888 	aRegion.Clear();
   889 	for(CWsClientWindow *win=Child();win;win=win->NextSibling())
   890 		aRegion.Union(*win->BaseArea());
   891 	}
   892 
   893 void CWsWindowGroup::SetOrdinalPosition(TInt aPos)
   894 	{
   895 	if (aPos==(TInt)KOrdinalPositionSwitchToOwningWindow)
   896 		SwitchToOwningWindow(NULL);
   897 	else
   898 		SetOrdinalPosition(aPos,NULL);
   899 	}
   900 
   901 TBool CWsWindowGroup::SetOrdinalPosition(TInt aPos,CWsWindowGroup* aClosingWindow)
   902 	{
   903 	TBool ret=ETrue;
   904 	
   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);	
   909 	
   910 	if (!iQueue)
   911 		ret=DoSetOrdinalPosition1(aPos,aClosingWindow);
   912 	else
   913 		{
   914 		TDblQueIter<CWsWindowGroup> iter(*iQueue);
   915 		CWsWindowGroup* group;
   916 		iter.SetToLast();
   917 		TInt after=0;
   918 		TInt before=0;
   919 		TInt* inc=&before;
   920 		while ((group=iter--)!=NULL)
   921 			{
   922 			if (group==this)
   923 				inc=&after;
   924 			++(*inc);
   925 			}
   926 		TInt lastWinGpPos=NumWindowGroupsOnMyScreen(OrdinalPriority())-after;
   927 		if (aPos<0)
   928 			aPos=lastWinGpPos;
   929 		else
   930 			aPos=Min(aPos,lastWinGpPos);
   931 		aPos-=before;
   932 		aPos=Max(aPos,0);
   933 		iter.SetToLast();
   934 		CWsWindowGroup* firstForward=iter--;
   935 		while (firstForward && firstForward->OrdinalPosition(EFalse)<aPos)
   936 			{
   937 			firstForward=iter--;
   938 			++aPos;
   939 			}
   940 		if (!firstForward)
   941 			iter.SetToFirst();
   942 		else
   943 			{
   944 			iter.Set(*firstForward);
   945 			MoveChainedWindows(iter,ETrue,aPos,aClosingWindow);
   946 			iter.Set(*firstForward);
   947 			iter++;
   948 			}
   949 		MoveChainedWindows(iter,EFalse,--aPos,aClosingWindow);
   950 #if defined(_DEBUG)
   951 		iter.SetToLast();
   952 		TInt pos1=-1;
   953 		TInt pos2;
   954 		TBool ok=ETrue;
   955 		while ((group=iter--)!=this)
   956 			{
   957 			pos2=group->OrdinalPosition(EFalse);
   958 			if (pos2<=pos1)
   959 				ok=EFalse;
   960 			pos1=pos2;
   961 			}
   962 		WS_ASSERT_DEBUG(ok, EWsPanicGroupWindowChainError);
   963 #endif
   964 		}
   965 
   966 	// If the ordinal positions have changed, check to see if there is a new render orientation
   967 	// and publish it if so
   968 	if(changed)
   969 	    CWsTop::CheckRenderOrientation();
   970 	
   971 	return ret;
   972 	}
   973 
   974 
   975 void CWsWindowGroup::MoveChainedWindows(TDblQueIter<CWsWindowGroup>& aIter,TBool aForward,TInt aPos,CWsWindowGroup* aClosingWindow)
   976 	{
   977 	CWsWindowGroup* groupWindow;
   978 	while ((groupWindow=(aForward ? aIter-- : aIter++))!=NULL)
   979 		{
   980 		groupWindow->DoSetOrdinalPosition1(aPos,aClosingWindow);
   981 		(aForward ? ++aPos : --aPos);
   982 		}
   983 	}
   984 
   985 TBool CWsWindowGroup::DoSetOrdinalPosition1(TInt aPos,CWsWindowGroup* aClosingWindow)
   986 	{
   987 	TBool ret=EFalse;
   988 	if (CheckOrdinalPositionChange(aPos))
   989 		{
   990 		if (Child())	// A group window with no children can not affect shadows
   991 			{
   992 			ret=ETrue;
   993 			}
   994 		DoSetOrdinalPosition2(aPos,aClosingWindow);
   995 		}
   996 	else if (aClosingWindow)    // do not reset focus if current groupwindow did not change its ordinal position
   997 		iScreen->ResetFocus(aClosingWindow);
   998 	return ret;
   999 	}
  1000 
  1001 void CWsWindowGroup::DoSetOrdinalPosition2(TInt aPos, CWsWindowGroup *aClosingWindow)
  1002 	{
  1003 	ChangeWindowPosition(aPos,iParent);
  1004 	ResetFocus(aClosingWindow);
  1005 	}
  1006 
  1007 void CWsWindowGroup::LostFocus()
  1008 	{
  1009 	iTextCursor.LostFocus();
  1010 	iWsOwner->UpdateWindowOrdinalPrioritys();
  1011 	if (iClientHandle!=0)
  1012 		QueueEvent(EEventFocusLost);
  1013 	TWalkWindowTreeFocusChanged wwt(EFalse);
  1014 	WalkWindowTree(wwt,EWalkChildren);
  1015 	iWsOwner->SetClientPriority();
  1016 	}
  1017 
  1018 void CWsWindowGroup::ReceivedFocus()
  1019 	{
  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)" 
  1026 #if defined(_DEBUG)
  1027 	if ((iEventQueueTest) && (++iSkipCount==5))
  1028 		{
  1029 		iSkipCount=0;
  1030 		}
  1031 	else
  1032 		{
  1033 		WsOwner()->EventQueue()->MoveToFront();
  1034 		}
  1035 #else
  1036 	WsOwner()->EventQueue()->MoveToFront();
  1037 #endif
  1038 
  1039 	QueueEvent(EEventFocusGained);
  1040 	TWalkWindowTreeFocusChanged wwt(ETrue);
  1041 	WalkWindowTree(wwt,EWalkChildren);
  1042 	iWsOwner->SetClientPriority();
  1043 	UpdateKeyClickState();
  1044 	}
  1045 
  1046 TInt CWsWindowGroup::NumWindowGroups(TBool aAllPriorities, TInt aPriority)
  1047 	{
  1048 	TInt count=0;
  1049 	TInt screenNo;
  1050 	for(screenNo=0;screenNo<CWsTop::NumberOfScreens();++screenNo)
  1051 		{
  1052 		count+=CWsWindowGroup::NumWindowGroupsOnScreen(CWsTop::Screen(screenNo)->RootWindow()->Child(),aAllPriorities,aPriority);
  1053 		}
  1054 	return(count);
  1055 	}
  1056 
  1057 TInt CWsWindowGroup::NumWindowGroupsOnScreen(const CWsWindowGroup* aGroupWin,TBool aAllPriorities,TInt aPriority)
  1058 	{
  1059 	TInt count=0;
  1060 	while (aGroupWin)
  1061 		{
  1062 		if (aAllPriorities || aGroupWin->iOrdinalPriority==aPriority)
  1063 			++count;
  1064 		aGroupWin=aGroupWin->NextSibling();
  1065 		}
  1066 	return count;
  1067 	}
  1068 
  1069 inline TInt CWsWindowGroup::NumWindowGroupsOnMyScreen(TInt aPriority)
  1070 	{
  1071 	return(CWsWindowGroup::NumWindowGroupsOnScreen(Parent()->Child(),EFalse,aPriority));
  1072 	}
  1073 
  1074 void CWsWindowGroup::GetFocusWindowGroupL(TInt aScreenNumber)
  1075 	{
  1076 	CWsWindowGroup *groupWin=(aScreenNumber==KDummyScreenNumber)?CWsTop::FocusWindowGroup():CWsTop::Screen(aScreenNumber)->FocusWindowGroup();	
  1077 	if (!groupWin)
  1078 		User::Leave(KErrGeneral);
  1079 	CWsClient::SetReply(groupWin->Identifier());
  1080 	}
  1081 
  1082 TInt CWsWindowGroup::GetWindowGroupListL(TInt aScreenNo,TBool aAllPriorities,TInt aPriority,TInt aCount,CArrayFixFlat<TInt>* aList)
  1083 	{
  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)
  1089 		{
  1090 		aList->AppendL(groupWin->Identifier());
  1091 		++count;
  1092 		groupWin=groupWin->NextSibling();
  1093 		}
  1094 	return count;
  1095 	}
  1096 	
  1097 TInt CWsWindowGroup::SendWindowGroupListL(TInt aScreenNumber, TBool aAllPriorities, TInt aPriority, TInt aCount)
  1098 	{
  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);
  1103 	TInt count(0);
  1104 	TInt requestedScreen=aScreenNumber;	
  1105 	if(requestedScreen==KDummyScreenNumber)
  1106 		{
  1107 		// get list from current focus screen first
  1108 		TInt focusScreenNo=CWsTop::CurrentFocusScreen()->ScreenNumber();
  1109 		count=GetWindowGroupListL(focusScreenNo, aAllPriorities, aPriority, aCount, list);
  1110 		if(count<aCount)
  1111 			{
  1112 			// now get from the remaining screen
  1113 			TInt screenNo;
  1114 			for(screenNo=0;screenNo<CWsTop::NumberOfScreens() && count<aCount;++screenNo)
  1115 				{
  1116 				// skip focus screen
  1117 				if (screenNo==focusScreenNo)
  1118 					continue;
  1119 				// count hold total number of window groups collected so far
  1120 				count=GetWindowGroupListL(screenNo, aAllPriorities, aPriority, aCount, list);
  1121 				}			
  1122 			}
  1123 		}
  1124 	else
  1125 		{
  1126 		count=GetWindowGroupListL(requestedScreen, aAllPriorities, aPriority, aCount, list);	
  1127 		}
  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
  1132 	}
  1133 
  1134 void CWsWindowGroup::GetWindowGroupListAndChainL(TInt aScreen,TBool aAllPriorities,TInt aPriority
  1135 																,RArray<RWsSession::TWindowGroupChainInfo>& list,TInt& aCountLeft)
  1136 	{
  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)
  1141 		{
  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);
  1147 		--aCountLeft;
  1148 		groupWin=groupWin->NextSibling();
  1149 		}
  1150 	}
  1151 
  1152 TInt CWsWindowGroup::SendWindowGroupListAndChainL(TBool aAllPriorities, TInt aPriority, TInt aCount)
  1153 	{
  1154 	if ((aCount<1) || (aCount>(KArrayMaxGranularity/sizeof(RWsSession::TWindowGroupChainInfo))))
  1155 		User::Leave(KErrArgument);
  1156 	RArray<RWsSession::TWindowGroupChainInfo> list(aCount);
  1157 	CleanupClosePushL(list);
  1158 	TInt count=aCount;
  1159 	TInt focusScreenNo=CWsTop::CurrentFocusScreen()->ScreenNumber();
  1160 	GetWindowGroupListAndChainL(focusScreenNo,aAllPriorities,aPriority,list,count);
  1161 	TInt screenNo;
  1162 	for(screenNo=0;screenNo<CWsTop::NumberOfScreens();++screenNo)
  1163 		{
  1164 		if (screenNo!=focusScreenNo)
  1165 			GetWindowGroupListAndChainL(screenNo,aAllPriorities,aPriority,list,count);
  1166 		}
  1167 	if (list.Count()>0)
  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
  1171 	}
  1172 
  1173 TBool CWsWindowGroup::SendEventToAllGroups(TBool aAllPriorities,TBool aOnePerClient,const TWsClCmdSendEventToWindowGroup& aData)
  1174 	{
  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;
  1180 	TInt screenNo;
  1181 	for(screenNo=0;screenNo<CWsTop::NumberOfScreens();++screenNo)
  1182 		{
  1183 		CWsWindowGroup *groupWin=CWsTop::Screen(screenNo)->RootWindow()->Child();
  1184 		if (!aAllPriorities)
  1185 			{
  1186 			while(groupWin && groupWin->iOrdinalPriority!=priority)
  1187 				groupWin=groupWin->NextSibling();
  1188 			}
  1189 		CWsWindowGroup* firstGroupWin=groupWin;
  1190 		CWsClient* lastOwner=NULL;
  1191 		CWsWindowGroup* groupWin2;
  1192 		while(groupWin && (aAllPriorities || groupWin->iOrdinalPriority==priority))
  1193 			{
  1194 			if (aOnePerClient)
  1195 				{
  1196 				if (lastOwner==groupWin->iWsOwner)
  1197 					goto ContinueLoop;
  1198 				lastOwner=groupWin->iWsOwner;
  1199 				for(groupWin2=firstGroupWin;groupWin2!=groupWin;groupWin2=groupWin2->NextSibling())
  1200 					{
  1201 					if (groupWin2->iWsOwner==groupWin->iWsOwner)
  1202 						break;
  1203 					}
  1204 				if (groupWin2->iWsOwner==groupWin->iWsOwner && groupWin2!=groupWin)
  1205 					goto ContinueLoop;
  1206 				}
  1207 			event.SetHandle(groupWin->ClientHandle());
  1208 			if (!groupWin->EventQueue()->QueueEvent(event))
  1209 				sentToAll=EFalse;
  1210 		ContinueLoop:
  1211 			groupWin=groupWin->NextSibling();
  1212 			}
  1213 		}
  1214 	return sentToAll;
  1215 	}
  1216 
  1217 void CWsWindowGroup::ReleasePendedMessagesToAllGroups(CWsClient * aClient)
  1218 	{
  1219 	TInt screenNo;
  1220 	for (screenNo = 0; screenNo < CWsTop::NumberOfScreens(); ++screenNo)
  1221 		{
  1222 		CWsWindowGroup* groupWin = CWsTop::Screen(screenNo)->RootWindow()->Child();
  1223 		while (groupWin)
  1224 			{
  1225 			if (groupWin->WsOwner() == aClient)
  1226 				{
  1227 				groupWin->ReleasePendedMessage();
  1228 				}
  1229 			groupWin = groupWin->NextSibling();
  1230 			}
  1231 		}
  1232 	}
  1233 
  1234 void CWsWindowGroup::ReleasePendedMessage()
  1235     {
  1236     if (iMessageArray->Count() > 0 && !(iFlags & EGroupFlagMessageSignalled))
  1237         {
  1238         if (!SignalMessageReady())
  1239 			{
  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)
  1243 			}
  1244         }
  1245     }
  1246 
  1247 void CWsWindowGroup::SendMessageToAllGroupsL(CWsClient& aSender,TBool aAllPriorities,const TWsClCmdSendMessageToWindowGroup& aData)
  1248 	{
  1249 	TInt screenNo;
  1250 	for(screenNo=0;screenNo<CWsTop::NumberOfScreens();++screenNo)
  1251 		{
  1252 		CWsWindowGroup* groupWin=CWsTop::Screen(screenNo)->RootWindow()->Child();		
  1253 		if (!aAllPriorities)
  1254 			{
  1255 			while(groupWin && groupWin->iOrdinalPriority!=aData.identifierOrPriority)
  1256 				groupWin=groupWin->NextSibling();
  1257 			}
  1258 		while(groupWin && (aAllPriorities || (groupWin->iOrdinalPriority==aData.identifierOrPriority)))
  1259 			{
  1260 			groupWin->QueueMessageL(aData.uid, aData.dataLength, aSender);
  1261 			groupWin=groupWin->NextSibling();
  1262 			}
  1263 		}
  1264 	}
  1265 
  1266 CWsWindowGroup *CWsWindowGroup::WindowGroupFromIdentifier(TInt aIdentifier)
  1267 	{
  1268 	// apply to all screens
  1269 	TInt screenNo;
  1270 	for (screenNo=0; screenNo<CWsTop::NumberOfScreens(); ++screenNo)
  1271 		{
  1272 		CWsWindowGroup* group;
  1273 		for(group=CWsTop::Screen(screenNo)->RootWindow()->Child(); group; group=group->NextSibling())
  1274 			{
  1275 			if (group->Identifier() == aIdentifier)
  1276 				return group;
  1277 			}
  1278 			
  1279 		}
  1280 		
  1281 	return NULL;
  1282 	}
  1283 
  1284 CWsWindowGroup *CWsWindowGroup::WindowGroupFromIdentifierL(TInt aIdentifier)
  1285 	{
  1286 	CWsWindowGroup *group=WindowGroupFromIdentifier(aIdentifier);
  1287 	if (!group)
  1288 		User::Leave(KErrNotFound);
  1289 	return(group);
  1290 	}
  1291 
  1292 CWsWindowGroup *CWsWindowGroup::FindWindowGroupL(CWsClient* aClient, TInt aIdentifier,TInt aOffset,const TPtrC *aMatch,const TThreadId *aThreadId)
  1293 	{
  1294 	CWsWindowGroup *group;
  1295 	if (aIdentifier)
  1296 		{
  1297 		group=WindowGroupFromIdentifier(aIdentifier);
  1298 		if (group)	// NULL group will cause KErrNotFound to be returned
  1299 			group=group->NextSibling();
  1300 		}
  1301 	else
  1302 		{
  1303 		// get window group for this session
  1304 		//
  1305 		group = aClient->Screen()->RootWindow()->Child();
  1306 		}
  1307 
  1308 	for(;group;group=group->NextSibling())
  1309 		{
  1310 		if (aThreadId)
  1311 			{
  1312 			if (group->WsOwner()->Client().Id()==*aThreadId)
  1313 				break;	// Found one
  1314 			}
  1315 		else
  1316 			{
  1317 			const TDesC *groupName=&nullDescriptor;
  1318 			if (group->GroupName())
  1319 				groupName=group->GroupName();
  1320 			if (groupName->Length()>=aOffset && groupName->Mid(aOffset).MatchF(*aMatch)>=0)
  1321 				break;	// Found one
  1322 			}
  1323 		}
  1324 	if (!group)
  1325 		User::Leave(KErrNotFound);
  1326 	return(group);
  1327 	}
  1328 
  1329 void CWsWindowGroup::AddPriorityKeyL(TUint aKeycode, TUint aModifierMask, TUint aModifiers)
  1330 	{
  1331 	iPriorityKeys=new(ELeave) TPriorityKey(aKeycode,aModifierMask,aModifiers,iPriorityKeys);
  1332 	}
  1333 
  1334 void CWsWindowGroup::RemovePriorityKey(TUint aKeycode, TUint aModifierMask, TUint aModifiers)
  1335 	{
  1336 	for(TPriorityKey **ppk=&iPriorityKeys;*ppk;ppk=&((*ppk)->iNext))
  1337 		if ((*ppk)->Equals(aKeycode, aModifierMask, aModifiers))
  1338 			{
  1339 			TPriorityKey *next=(*ppk)->iNext;
  1340 			delete *ppk;
  1341 			*ppk=next;
  1342 			break;
  1343 			}
  1344 	}
  1345 
  1346 void CWsWindowGroup::RemoveAllPriorityKeys()
  1347 	{
  1348 	TPriorityKey *pk=iPriorityKeys;
  1349 	while(pk)
  1350 		{
  1351 		TPriorityKey *next=pk->iNext;
  1352 		delete pk;
  1353 		pk=next;
  1354 		}
  1355 	}
  1356 
  1357 TBool CWsWindowGroup::CheckForPriorityKey(const TKeyEvent &aKeyEvent)
  1358 	{
  1359 	for(TPriorityKey *pk=iPriorityKeys;pk;pk=pk->iNext)
  1360 		{
  1361 		if (pk->KeyMatches(aKeyEvent))
  1362 			{	
  1363 			WsOwner()->PriorityKeyPressed(ClientHandle(), aKeyEvent);
  1364 			return(ETrue);
  1365 			}
  1366 		}
  1367 	return(EFalse);
  1368 	}
  1369 
  1370 void CWsWindowGroup::StatusDump(TDes &aBuf)
  1371 	{
  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);
  1374 	}
  1375 
  1376 TBool CWsWindowGroup::SignalMessageReady()
  1377 	{
  1378 	TWsEvent event;
  1379 	event.SetType(EEventMessageReady);
  1380 	event.SetHandle(ClientHandle());
  1381 	event.SetTimeNow();
  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);
  1387 	if (result)
  1388 		{
  1389 		iFlags |= EGroupFlagMessageSignalled;
  1390 		}
  1391 	return result;
  1392 	}
  1393 
  1394 void CWsWindowGroup::QueueMessageL(TUid aUid, TInt aDataLength, CWsClient& aSender)
  1395 	{
  1396 	WS_ASSERT_DEBUG(iFlags&(EGroupFlagMsgQueueActive|EGroupFlagMsgQueueNew) || iMessageArray->Count()>=1,EWsPanicMsgQueueError);
  1397 	if (!(iFlags&(EGroupFlagMsgQueueActive|EGroupFlagMsgQueueNew)) && iMessageArray->Count()>=KMaxNumberOfMsgsInInactiveQueue)
  1398 		{
  1399 		WS_ASSERT_DEBUG(iMessageArray->Count()<=KMaxNumberOfMsgsInInactiveQueue,EWsPanicMsgQueueError);
  1400 		iMessageArray->Delete(1,iMessageArray->Count()-1);
  1401 		}
  1402 	TWsMessage* message=NULL;
  1403 	TRAPD(err,message=&iMessageArray->ExtendL(aDataLength+sizeof(aUid)));
  1404 	if ((err || (iFlags&EGroupFlagMsgQueueNew)) && iMessageArray->Count()>KMaxNumberOfMsgsInQueue)
  1405 		{
  1406 		iFlags&=~(EGroupFlagMsgQueueActive|EGroupFlagMsgQueueNew);
  1407 		iMessageArray->Delete(1,iMessageArray->Count()-(err?1:2));
  1408 		iMessageArray->Compress();
  1409 		}
  1410 	User::LeaveIfError(err);
  1411 	if (message)
  1412 		{
  1413 		message->iUid=aUid;
  1414 		TPtr8 ptr(&message->iTheRest[0],aDataLength);
  1415 		TRAP(err,aSender.RemoteReadL(ptr,0));
  1416 		if (err)
  1417 			{
  1418 			iMessageArray->Delete(iMessageArray->Count()-1);
  1419 			User::Leave(err);
  1420 			}
  1421 		if (!(iFlags & EGroupFlagMessageSignalled))
  1422 			{
  1423 			if (!SignalMessageReady())
  1424 				{
  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)
  1428 				}
  1429 			}
  1430 		}
  1431 	}
  1432 
  1433 void CWsWindowGroup::FetchMessageL()
  1434 	{
  1435 	if (!(iFlags & EGroupFlagMessageSignalled))
  1436 		{
  1437 		OwnerPanic(EWservPanicFetchMessage);
  1438 		}
  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)
  1444 		{
  1445 		if (!SignalMessageReady())
  1446 			{
  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)
  1450 			}
  1451 		}
  1452 	}
  1453 
  1454 TBool CWsWindowGroup::ScreenDeviceValid() const
  1455 	{
  1456 	return(iScreenDevice?iScreenDevice->ScreenDeviceValidState():(iScreen->ScreenSizeMode()==0));
  1457 	}
  1458 
  1459 TBool CWsWindowGroup::CanReceiveFocus() const
  1460 	{
  1461 	return(ReceivesFocus() && iWsOwner->NotClosing() && (ScreenDeviceValid() || iFlags&EGroupFlagHandlesDeviceChange));
  1462 	}
  1463 
  1464 void CWsWindowGroup::SetScreenChangeEventStateL(TBool aEnabled)
  1465 	{
  1466 	iFlags&=~EGroupFlagHandlesDeviceChange;
  1467 	if (aEnabled)
  1468 		{
  1469 		iFlags|=EGroupFlagHandlesDeviceChange;
  1470 		TWindowServerEvent::AddToScreenDeviceChangeEventListL(*this);
  1471 		if (iScreen->ScreenSizeMode()!=0)
  1472 			TWindowServerEvent::SendScreenDeviceChangedEvent(this);
  1473 		}
  1474 	else
  1475 		TWindowServerEvent::RemoveFromScreenDeviceChangeEventList(*this);
  1476 	iScreen->ResetFocus(NULL);
  1477 	}
  1478 
  1479 void CWsWindowGroup::SetScreenDeviceValidState(const DWsScreenDevice *aDevice)
  1480 	{
  1481 	if (iScreenDevice==aDevice)
  1482 		{
  1483 		TBool state=ScreenDeviceValid();
  1484 		for(CWsTopClientWindow *win=Child();win;win=win->NextSiblingTop())
  1485 			{
  1486 			win->SetScreenDeviceValidState(state);
  1487 			}
  1488 		}
  1489 	}
  1490 
  1491 void CWsWindowGroup::SetScreenDeviceValidStates(const DWsScreenDevice *aDevice)
  1492 	{
  1493 	for(CWsWindowGroup *groupWin=aDevice->RootWindow()->Child();groupWin;groupWin=groupWin->NextSibling())
  1494 		groupWin->SetScreenDeviceValidState(aDevice);
  1495 	}
  1496 
  1497 void CWsWindowGroup::SetScreenDeviceValidStates(CScreen* aScreen)
  1498 	{
  1499 	CWsRootWindow* rootWindow = aScreen->RootWindow();
  1500 
  1501 	CWsWindowGroup* groupWin;
  1502 	CWsTopClientWindow* win;
  1503 	for(groupWin=rootWindow->Child();groupWin;groupWin=groupWin->NextSibling())
  1504 		{
  1505 		TBool state=groupWin->ScreenDeviceValid();
  1506 		for(win=groupWin->Child();win;win=win->NextSiblingTop())
  1507 			{
  1508 			win->SetScreenDeviceValidStateFlag(state);
  1509 			win->ResetHiddenFlagsInParentAndChildren();
  1510 			}
  1511 		}
  1512 	}
  1513 
  1514 void CWsWindowGroup::SetScreenDevice(DWsScreenDevice *aDevice)
  1515 	{
  1516 	iScreenDevice=aDevice;	
  1517 	}
  1518 
  1519 void CWsWindowGroup::NewOrientation(TInt aMode,CFbsBitGc::TGraphicsOrientation aRotation, CWsRootWindow* aRootWindow)
  1520 	{
  1521 	for(CWsWindowGroup *groupWin=aRootWindow->Child();groupWin;groupWin=groupWin->NextSibling())
  1522 		{
  1523 		DWsScreenDevice *device=groupWin->Device();
  1524 		if (device)
  1525 			device->NewOrientation(aMode,aRotation);
  1526 		}
  1527 	}
  1528 
  1529 void CWsWindowGroup::ResetFocus(CWsWindowGroup *aClosingWindow)
  1530 	{
  1531 	if (iScreen)
  1532 		{
  1533 		iScreen->ResetFocus(aClosingWindow);
  1534 		}
  1535 	}
  1536 
  1537 TBool CWsWindowGroup::IsChained(TInt& aParentId)
  1538 	{
  1539 	if (!iQueue)
  1540 		return EFalse;
  1541 	if (iQueue->First()==this)
  1542 		aParentId=0;
  1543 	else
  1544 		aParentId=BeforeInChain()->Identifier();
  1545 	return ETrue;
  1546 	}
  1547 
  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)));
  1551 	}
  1552 
  1553 TBool CWsWindowGroup::CheckCapability(TInt& aOrdinalPriority)
  1554 	{
  1555 	if(aOrdinalPriority>=KPasswordWindowGroupPriority)
  1556 		{
  1557 		if(!KSecurityPolicy_SwEvent().CheckPolicy(WsOwner()->Client(),__PLATSEC_DIAGNOSTIC_STRING("Capability check failed")))
  1558 			{
  1559 			aOrdinalPriority=KPasswordWindowGroupPriority-1;
  1560 			return EFalse;
  1561 			}
  1562 		}
  1563  	return ETrue;
  1564 	}
  1565 	
  1566 TBool CWsWindowGroup::HasVisibleTranslucentChild()
  1567 	{
  1568 	CWsWindowBase * child = iChild;
  1569 	while (child)
  1570 		{
  1571 		if (child->WinType() == EWinTypeClient)
  1572 			{
  1573 			CWsClientWindow * cliwin = static_cast<CWsClientWindow *>(child);
  1574 			if (cliwin->IsTranslucent() && cliwin->IsVisible())
  1575 				return ETrue;
  1576 			}
  1577 		else if (static_cast<CWsWindowGroup*>(child)->HasVisibleTranslucentChild())
  1578 			{
  1579 			return ETrue;
  1580 			}
  1581 		child = child->NextSibling();
  1582 		}
  1583 	return EFalse;
  1584 	}
  1585 
  1586 
  1587 TInt CWsWindowGroup::NumClientWindowGroups()
  1588 	{
  1589 	CWsObjectIx& objix=*WsOwner()->ObjectIndex();
  1590 	const TWsObject* ptr=objix.FirstObject();
  1591 	const TWsObject* end=ptr+objix.Length();
  1592 	TInt count=0;
  1593 	while(++ptr<end)		//First one should always have a NULL object
  1594 		{
  1595 		const CWsObject* obj=ptr->iObject;
  1596 		if (obj && obj->Type()==WS_HANDLE_GROUP_WINDOW)
  1597 			++count;
  1598 		}
  1599 	return count;
  1600 	}
  1601 	
  1602 void CWsWindowGroup::SetEventQueueTestState(TBool aEventQueState)
  1603 	{
  1604 	iEventQueueTest = aEventQueState;
  1605 	}