os/graphics/windowing/windowserver/nga/SERVER/openwfc/cliwin.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-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Client window functions
    15 // 
    16 //
    17 
    18 #include "W32CLICK.H"
    19 #include <graphics/wselement.h>
    20 #include "server.h"
    21 #include "cliwin.h"
    22 #include "gc.h"
    23 #include "rootwin.h"
    24 #include "windowgroup.h"
    25 #include "walkwindowtree.h"
    26 #include "ScrDev.H"
    27 #include "wstop.h"
    28 #include "EVQUEUE.H"
    29 #include "KEYCLICK.H"
    30 #include "panics.h"
    31 #include "password.h"
    32 #include "pointer.h"
    33 #include "EVENT.H"
    34 #include "backedupwindow.h"
    35 #include "redrawmsgwindow.h"
    36 #include "ANIM.H"
    37 
    38 #include "windowelementset.h"
    39 
    40 
    41 
    42 
    43 TBool CWsClientWindow::iAbsoluteFading = EFalse;
    44 
    45 const TPoint corner1[1]={TPoint(1,1)};
    46 const TPoint corner2[2]={TPoint(2,1),TPoint(1,1)};
    47 const TPoint corner3[2]={TPoint(3,1),TPoint(1,2)};
    48 const TPoint corner5[4]={TPoint(5,1),TPoint(3,1),TPoint(2,1),TPoint(1,2)};
    49 
    50 /*CWsClientWindow*/
    51 
    52 CWsClientWindow::CWsClientWindow(CWsClient* aOwner, CScreen* aScreen) : CWsWindow(aOwner,WS_HANDLE_WINDOW,aScreen)
    53 	{
    54 	iWinType=EWinTypeClient;
    55 	}
    56 
    57 void CWsClientWindow::ConstructL(const TWsClCmdCreateWindow &aCmd, CWsWindowBase *aParent, TBool aScreenDeviceIsInvalid)
    58 	{
    59 	CWsWindow::Construct();
    60 	NewObjL();
    61 	if (aCmd.clientHandle==NULL)
    62 		OwnerPanic(EWservPanicNullHandle);
    63 #if defined(_DEBUG)
    64 	if (IsClientHandleInUse(aCmd.clientHandle))
    65 		OwnerPanic(EWservPanicDuplicateHandle);
    66 #endif
    67 	iPointerFilter=EPointerFilterEnterExit|EPointerFilterMove|EPointerFilterDrag;
    68 	iClientHandle=aCmd.clientHandle;
    69 	CWsWindow* inherit=static_cast<CWsWindow *>(aParent);
    70 	if (aParent->WinType()==EWinTypeGroup)
    71 		inherit=RootWindow();
    72 	SetPointerCursor(aParent->PointerCursor());
    73 	iAbs=inherit->Abs();
    74 	iOrigin=aParent->Origin();
    75 	iRel.iBr.iX=inherit->Rel().iBr.iX-inherit->Rel().iTl.iX;
    76 	iRel.iBr.iY=inherit->Rel().iBr.iY-inherit->Rel().iTl.iY;
    77 	switch(aCmd.type)
    78 		{
    79 		case EWinRedraw:
    80 			iRedraw=new(ELeave) CWsRedrawMsgWindow(this);
    81 			break;
    82 		case EWinBackedUp:
    83 			iRedraw=new(ELeave) CWsBackedUpWindow(this, aCmd.displayMode);
    84 			iAbs.iBr=iAbs.iTl;
    85 			iRel.iBr=iRel.iTl;
    86 			break;
    87 		case EWinBlank:
    88 			iRedraw=new(ELeave) CWsBlankWindow(this);
    89 			break;
    90 		default:
    91 			OwnerPanic(EWservPanicRedrawType);
    92 		}
    93 	ResetHiddenFlag();
    94 	SetCornerTypeL(EWindowCornerSquare,0,NULL,EFalse);
    95 	CWsWindowBase::ConstructL(aParent);
    96 	if (aScreenDeviceIsInvalid)
    97 		{
    98 		iFlags|=EFlagScreenDeviceInvalid;
    99 		ResetHiddenFlag();
   100 		}
   101 	iRedraw->ConstructL();
   102 	}
   103 
   104 void CWsClientWindow::GetBaseAreaOfNode(RWsRegion &aRegion) const
   105     {
   106     if (iBaseArea)
   107         {
   108         aRegion.Copy(*iBaseArea);
   109         }
   110     aRegion.ClipRect(iAbs);
   111     }
   112 
   113 void CWsClientWindow::ClipRegionToBaseArea(RWsRegion &aRegion) const
   114     {
   115     if (iBaseArea)
   116         {
   117         aRegion.Intersect(*iBaseArea);
   118         }
   119     aRegion.ClipRect(iAbs);
   120     }
   121 
   122 void CWsClientWindow::GetClippedBaseArea(RWsRegion &aRegion) const
   123 	{
   124 	const CWsWindowBase* ancestor = BaseParent();	    
   125 	GetBaseAreaOfNode(aRegion);	
   126 	while (ancestor && ancestor->WinType() == EWinTypeClient)
   127 	    {
   128 	    static_cast<const CWsClientWindow*>(ancestor)->ClipRegionToBaseArea(aRegion);       
   129 	    ancestor = ancestor->BaseParent();
   130 	    }
   131 	}
   132 
   133 void CWsClientWindow::GetOpaqueBaseAreaOfNode(RWsRegion &aRegion) const
   134     {
   135     if (IsTranslucent())
   136         {
   137         if (iUserDefinedOpaqueRegion)
   138             {
   139             aRegion.Copy(*iUserDefinedOpaqueRegion);
   140             aRegion.ClipRect(iAbs);
   141             }
   142         else
   143             {
   144             aRegion.Clear();
   145             }
   146         }
   147     else
   148         {
   149         GetBaseAreaOfNode(aRegion);
   150         }
   151     }
   152 
   153 void CWsClientWindow::ClipRegionToOpaqueBaseArea(RWsRegion& aRegion) const
   154     {
   155     if (IsTranslucent())
   156         {
   157         if (iUserDefinedOpaqueRegion)
   158             {
   159             aRegion.Intersect(*iUserDefinedOpaqueRegion);
   160             aRegion.ClipRect(iAbs);
   161             }
   162         else
   163             {
   164             aRegion.Clear();
   165             }
   166         }
   167     else
   168         {
   169         ClipRegionToBaseArea(aRegion);
   170         }
   171     }
   172 
   173 void CWsClientWindow::GetOpaqueClippedBaseArea(RWsRegion &aRegion) const
   174 	{
   175 	const CWsWindowBase* ancestor = BaseParent();
   176 	GetOpaqueBaseAreaOfNode(aRegion);
   177 	while (ancestor && ancestor->WinType() == EWinTypeClient)
   178 	    {
   179 	    static_cast<const CWsClientWindow*>(ancestor)->ClipRegionToOpaqueBaseArea(aRegion);
   180 	    ancestor = ancestor->BaseParent();
   181 	    }
   182 	}
   183 
   184 TInt CWsClientWindow::GetNonOpaqueBaseAreaOfNode(RWsRegion &aRegion) const
   185     {
   186     aRegion.Clear();
   187     if (IsTranslucent())
   188         {
   189         if(iUserDefinedTransparentRegion)
   190             {
   191             aRegion.Copy(*iUserDefinedTransparentRegion);
   192             aRegion.ClipRect(iAbs);
   193             return KErrNone;
   194             }
   195         }  
   196     return KErrNotFound;
   197     }
   198 
   199 void CWsClientWindow::ResetHiddenFlag()
   200 //
   201 // Reset the status of the hidden flag based on the current states of the active and invisible flags
   202 //
   203 	{
   204 	CWsClientWindow *parent=static_cast<CWsClientWindow*>(iParent);
   205 
   206 	TBool wasHidden = iFlags&EFlagHidden;
   207 	TBool nowHidden = (parent==NULL || 
   208 		              (parent->WinType()==EWinTypeClient && !parent->IsVisible()) ||
   209 	                  !(iFlags&EFlagActive) ||
   210 	                  (iFlags&EFlagInvisible) ||
   211 			          (iFlags&EFlagScreenDeviceInvalid));
   212 
   213 	if (nowHidden)
   214 		{
   215 		iFlags|=EFlagHidden;
   216 		iFlags&=~EFlagDrawnToScreen;
   217 		}
   218 	else
   219 		{
   220 		iFlags&=~EFlagHidden;
   221 		}
   222 	if ((!nowHidden) != (!wasHidden))
   223 		{
   224 		// intentionally call the screen directly
   225 		iScreen->ScheduleRegionUpdate(&iVisibleRegion);
   226 		
   227 		WS_ASSERT_DEBUG(!iDirtyWindowRegion.CheckError(),EWsPanicErrorInRegion);
   228 		WS_ASSERT_DEBUG(!iDirtySpriteRegion.CheckError(),EWsPanicErrorInRegion);
   229 	//	WS_ASSERT_DEBUG(!InvalidArea().CheckError(),EWsPanicErrorInRegion);		//error flag in invalid area may be set in OOM cases
   230 																				//thus the assert statement is caused to fail
   231 		if (wasHidden && iScreen->ChangeTracking() && 
   232 				(!iDirtyWindowRegion.IsEmpty() || !iDirtySpriteRegion.IsEmpty() || !InvalidArea().IsEmpty()))
   233 			{
   234 			// Window has just become visible, schedule it
   235 			iScreen->ScheduleWindow(this);
   236 			}
   237 		}
   238 	}
   239 	
   240 void CWsClientWindow::ResetHiddenFlags()
   241 	{
   242 	CWsClientWindow *win=this;
   243 	FOREVER
   244 		{
   245 		TUint oldHiddenFlag=win->iFlags&EFlagHidden;
   246 		win->ResetHiddenFlag();
   247 		if ((win->iFlags&EFlagHidden)!=oldHiddenFlag)	// If hidden status hasn't changed nothing to do
   248 			{
   249 			win->SetElementOpacity(IsVisible() ? 0xFF : 0x00);	// Update element visibility
   250 			if (win->Child())
   251 				{
   252 				win=win->Child();
   253 				continue;
   254 				}
   255 			}
   256 		if (win==this)
   257 			return;
   258 		while(!win->NextSibling())
   259 			{
   260 			win=(CWsClientWindow *)win->BaseParent();
   261 			if (win==this)
   262 				return;
   263 			}
   264 		win=win->NextSibling();
   265 		} // for loop ends
   266 	}
   267 
   268 void CWsClientWindow::OffsetBaseArea(const TPoint &aOffset)
   269 	{
   270 	iBaseArea->Offset(aOffset);
   271 	//If the given window's position changes, then update.
   272 	if (aOffset.iX || aOffset.iY)
   273 		{
   274 		UpdateElementExtent(&aOffset);
   275 		}
   276 	}
   277 
   278 void CWsClientWindow::CalcBaseArea()
   279 //
   280 // The windows basic area before any clipping is done
   281 //
   282 	{
   283 	TInt cornerType=iCornerData&ECornerTypeMask;
   284 	if (cornerType==EWindowCornerRegion)
   285 		iBaseArea->ClipRect(FullRect());
   286 	else
   287 		{
   288 		TSize size=Size();
   289 		iBaseArea->Clear();
   290 		const TPoint *corners=NULL;
   291 		TInt count=0;
   292 		switch(cornerType)
   293 			{
   294 			case EWindowCorner1:
   295 				count=sizeof(corner1)/sizeof(TPoint);
   296 				corners=corner1;
   297 				break;
   298 			case EWindowCorner2:
   299 				count=sizeof(corner2)/sizeof(TPoint);
   300 				corners=corner2;
   301 				break;
   302 			case EWindowCorner3:
   303 				count=sizeof(corner3)/sizeof(TPoint);
   304 				corners=corner3;
   305 				break;
   306 			case EWindowCorner5:
   307 				count=sizeof(corner5)/sizeof(TPoint);
   308 				corners=corner5;
   309 				break;
   310 			default:
   311 				break;
   312 			}
   313 		TInt top=0;
   314 		TInt bot=size.iHeight;
   315 		for(TInt index=0;index<count;index++)
   316 			{
   317 			TInt xadjust=corners[index].iX;
   318 			TInt yadjust=corners[index].iY;
   319 			if ((iCornerData&(EWindowCornerNotTL|EWindowCornerNotTR))!=(EWindowCornerNotTL|EWindowCornerNotTR))
   320 				{
   321 				iBaseArea->AddRect(TRect(iCornerData&EWindowCornerNotTL?0:xadjust,top,
   322 										 size.iWidth-(iCornerData&EWindowCornerNotTR?0:xadjust),top+yadjust));
   323 				top+=yadjust;
   324 				}
   325 			if ((iCornerData&(EWindowCornerNotBL|EWindowCornerNotBR))!=(EWindowCornerNotBL|EWindowCornerNotBR))
   326 				{
   327 				iBaseArea->AddRect(TRect(iCornerData&EWindowCornerNotBL?0:xadjust,bot-yadjust,
   328 										 size.iWidth-(iCornerData&EWindowCornerNotBR?0:xadjust),bot));
   329 				bot-=yadjust;
   330 				}
   331 			}
   332 		iBaseArea->AddRect(TRect(0,top,size.iWidth,bot));
   333 		iBaseArea->Offset(Origin());
   334 		iBaseArea->ClipRect(FullRect());
   335 		iBaseArea->Sort();
   336 		}
   337 	}
   338 
   339 void CWsClientWindow::GenerateArea(RWsRegion &aArea, TBool aClipTranslucent) const
   340 //
   341 // Create the window area list.
   342 //
   343 	{
   344 	aArea.Clear();
   345 	if (IsVisible())
   346 		{
   347 		aArea.Copy(*iBaseArea);
   348 		aArea.ClipRect(iAbs);
   349 		const CWsClientWindow *win=this;
   350 		FOREVER
   351 			{
   352 			if (win->IsTopClientWindow())
   353 				break;
   354 			ClipWindows(aArea,(CWsClientWindow *)win->BaseParent()->BaseChild(),win,aClipTranslucent);
   355 			win=(CWsClientWindow *)win->iParent;
   356 			}
   357 		TInt tidyCount=0;
   358 		for(const CWsTopClientWindow *cwin=RootWindow()->FirstTopClientWindow();aArea.Count() && cwin!=win;cwin=cwin->NextSiblingMultiParent())
   359 			{
   360 			if (!tidyCount--)
   361 				{
   362 				aArea.Tidy();
   363 				tidyCount=ETidyCountSetting;	// Tidy every ETidyCountSetting times around
   364 				}
   365 			if (cwin->IsVisible())
   366 				{
   367 				if (cwin->IsTranslucent() && !aClipTranslucent)
   368 					{
   369 					if (cwin->iUserDefinedOpaqueRegion)
   370 						{
   371 						aArea.SubRegion(*cwin->iUserDefinedOpaqueRegion);
   372 						}
   373 					}
   374 				else
   375 					{
   376 					aArea.SubRegion(*cwin->iBaseArea);
   377 					}
   378 				}
   379 			}
   380 		aArea.Tidy();
   381 		}
   382 	}
   383 
   384 void CWsClientWindow::ClipWindows(TRegion &region,const CWsClientWindow *start, const CWsClientWindow *end, TBool aClipTranslucent)
   385 //
   386 // Remove out of the region the opaque part of the abs rect of all the windows starting from 'start'
   387 // along the sibling list to (and not including) the end window.
   388 //
   389 	{
   390 	for(const CWsClientWindow *win=start;region.Count() && win!=end;win=win->NextSibling())
   391 		{
   392 		if (win->IsVisible())
   393 			{
   394 			if (win->IsTranslucent() && !aClipTranslucent)
   395 				{
   396 				if (win->iUserDefinedOpaqueRegion)
   397 					{
   398 					region.SubRegion(*win->iUserDefinedOpaqueRegion);
   399 					}
   400 				}
   401 			else
   402 				{
   403 				region.SubRegion(*win->iBaseArea);
   404 				}
   405 			}
   406 		}
   407 	}
   408 
   409 void CWsClientWindow::GenerateTopRegion(RWsRegion& aRegion) const
   410 	{
   411 	GenerateArea(aRegion,ETrue);
   412 	if (iChild)
   413 		ClipWindows(aRegion,Child(),NULL,ETrue);
   414 	}
   415 
   416 void CWsClientWindow::GenerateWindowRegion(RWsRegion &aRegion) const
   417 //
   418 // Calculate the windows clipping region without using the usual stored iArea or iRegion fields
   419 // this function is used by the screen backup code to calculate "what if" regions to work out
   420 // whether something would be visible if the backed up window didn't exist, on this basis we
   421 // don't want to modify the existing copies of iArea & iRegion.
   422 //
   423 	{
   424 	GenerateArea(aRegion,EFalse);
   425 	if (iChild)
   426 		ClipWindows(aRegion,Child(),NULL,EFalse);
   427 	}
   428 
   429 const TRegion *CWsClientWindow::DrawingRegion()
   430 	{
   431 	return (&iRedraw->BaseDrawRegion());
   432 	}
   433 
   434 void CWsClientWindow::RecalcChildAbs(const TPoint *aOffset)
   435 	{
   436 	CWsClientWindow *win=this;
   437 	FOREVER
   438 		{
   439 		FOREVER
   440 			{
   441 			win->SetAbsFromRel();
   442 			if (aOffset)
   443 				win->OffsetBaseArea(*aOffset);
   444 			if (win->Child()==NULL)
   445 				break;
   446 			win=win->Child();
   447 			}
   448 		FOREVER
   449 			{
   450 			if (win==this)
   451 				return;
   452 			if (win->NextSibling()!=NULL)
   453 				{
   454 				win=win->NextSibling();
   455 				break;
   456 				}
   457 			win=(CWsClientWindow *)win->iParent;	// The cast is safe as the loop is aborted when win==this
   458 			}
   459 		}
   460 	}
   461 
   462 void CWsClientWindow::SetAbsFromRel()
   463 	{
   464 	iOrigin=iRel.iTl+iParent->Origin();
   465 	iAbs=iRel;
   466 	iAbs.Move(iParent->Origin());
   467 	iAbs.Intersection(iParent->AbsRect());
   468 	}
   469 
   470 void CWsClientWindow::SetExtentL(const TPoint *aPos,const TSize *aSize)
   471 	{
   472 	if (iParent==NULL)
   473 		OwnerPanic(EWservPanicParentDeleted);
   474 	TPoint offset = TPoint(0,0);
   475 	TSize oldSize;
   476 	TSize newSize;
   477 	TBool sizeChanged = EFalse;
   478 	TBool posChanged = EFalse;
   479 	
   480 	if (aPos)
   481 		{
   482 		offset = *aPos+iParent->Origin()-iOrigin;
   483 		if (offset.iX != 0 || offset.iY != 0)
   484 			{
   485 			posChanged = ETrue;
   486 			}
   487 		}
   488 		
   489 	if (posChanged)
   490 		{
   491 		TWalkWindowTreeScheduleRedraws wwt;
   492 		WalkWindowTree(wwt, EWalkChildren);
   493 		}
   494 	
   495 	if (aSize)
   496 		{
   497 		newSize=*aSize;
   498 		if (newSize.iWidth<0)
   499 			newSize.iWidth=0;
   500 		if (newSize.iHeight<0)
   501 			newSize.iHeight=0;
   502 		// This should be the only part of resizing that can fail
   503 		// and it can only fail for backedup windows.
   504 		iRedraw->PrepareForResizeL(newSize,oldSize);
   505 		sizeChanged = *aSize != iRel.Size();
   506 		}
   507 
   508 	if (posChanged)
   509 		{
   510 		iRel.Move(offset);
   511 		RecalcChildAbs(&offset);      // Also calls UpdateElementExtent(offset)
   512 		TWalkWindowTreeOffsetTransparentRegions offsetTransparent(offset);
   513 		WalkWindowTree(offsetTransparent, EWalkChildren);
   514 		}
   515 
   516 	if (sizeChanged)
   517 		{
   518 		iRel.SetSize(newSize);
   519 		RecalcChildAbs(NULL);
   520 		CalcBaseArea();
   521 		iRedraw->Resize(newSize,oldSize);
   522 		if (Redraw()->HasElement())
   523 		    UpdateElementExtent();
   524 		}
   525 	
   526 	if ((posChanged || sizeChanged) && Redraw()->HasElement())
   527 		{
   528 		TRect interSection(iParent->Origin(), iParent->Size());
   529 		interSection.Intersection(FullRect());
   530 		if (interSection == FullRect())
   531 			{
   532 			// There is no any clipping in this case
   533 			interSection = TRect();
   534 			}
   535 		// Get the corresponding source rectangle for the element
   536 		if (!iOriginalSrcElementRect.IsEmpty() && !iOriginalDestElementRect.IsEmpty() && !iAbs.IsEmpty())
   537 			{
   538 			MWsElement* element = Screen()->WindowElements().GetElementFromWindow(*this);
   539 			if (element)
   540 				{
   541 				element->SetDestinationClippingRect(interSection);
   542 				}
   543 			}
   544 		}	
   545 	
   546 	if (posChanged || sizeChanged)
   547 		{
   548 		iRedraw->ClipInvalidRegion(TRect(iRel.Size()));
   549 		iRedraw->Moved();
   550 		ScheduleRegionUpdate(NULL);
   551 		TWalkWindowTreeRecalcOpaque recalcOpaque;
   552 		WalkWindowTree(recalcOpaque, EWalkChildren);
   553 
   554 		MWsWindowTreeObserver* windowTreeObserver = Screen()->WindowTreeObserver();
   555 		if (windowTreeObserver)
   556 			{
   557 			TRect rect = FullRect();
   558 			windowTreeObserver->NodeExtentChanged(*this, rect);
   559 			
   560 			for (CWsAnim* anim = iAnimList; anim; anim = anim->Next())
   561 				{
   562 				windowTreeObserver->NodeExtentChanged(*anim, rect);
   563 				}
   564 			}
   565 		}
   566 	}
   567 
   568 
   569 void CWsClientWindow::GetElementFlipAndRotation(TBool& aElemetFlip, MWsElement::TElementRotation& aElemenetRotation)
   570     {
   571     MWsElement* element = Screen()->WindowElements().GetElementFromWindow(*this);
   572     if (element)
   573         {
   574         aElemetFlip = element->SourceFlipping();
   575         aElemenetRotation = element->SourceRotation();
   576         }
   577     }
   578 
   579 void CWsClientWindow::Scroll(const TRect &aClipRect, const TPoint &aOffset, const TRect &aRect)
   580 	{
   581 	if (iParent==NULL)
   582 		OwnerPanic(EWservPanicParentDeleted);
   583 	
   584 //
   585 	iRedraw->Scroll(aClipRect, aOffset,aRect);
   586 //
   587 	CWsTop::TriggerRedraws(RootWindow());
   588 	}
   589 
   590 void CWsClientWindow::DeleteBaseArea()
   591 	{
   592 	WS_ASSERT_DEBUG(iBaseArea!=&nullRegion, EWsPanicRegionNull);
   593  	if ((iCornerData&ECornerTypeMask)==EWindowCornerRegion)
   594 		((RWsRegion *)iBaseArea)->Destroy();
   595 	else
   596 		{
   597 		delete iBaseArea;
   598 		}
   599  	iBaseArea=NULL;
   600 	}
   601 
   602 CWsClientWindow::~CWsClientWindow()
   603 	{
   604 	while(iVisibleRegionTrackingCounter>0)
   605 		{
   606 		SetupVisibleRegionTracking(EFalse);
   607 		}
   608 	if (iBaseWinFlags&EBaseWinNodeCreated)
   609 		{
   610 		MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
   611 		if (windowTreeObserver)
   612 			{
   613 			windowTreeObserver->NodeReleased(*this);
   614 			iBaseWinFlags &= ~EBaseWinNodeCreated;
   615 			}
   616 		}
   617 	Shutdown();
   618 	SetUserTransparentRegion(0);
   619 	CWsPassword::WindowDestroyed(this);
   620 	}
   621 
   622 void CWsClientWindow::Shutdown()
   623 //
   624 // Destroy a window, disconnects from the window tree and destroys all it's child windows
   625 //
   626 	{
   627 	iFlags|=EFlagShutDownInProgress;
   628 	if (CClick::IsHandler())
   629 		{
   630 		TWindowCloseData params;
   631 		params.iClientHandle=iClientHandle;
   632 		//if parent already shutdown (or disconnected) send 0
   633 		params.iWindowGroupId = (iParent) ? WinGroup()->Identifier() : 0;
   634 		CClick::OtherEvent(EEventWindowClose,&params);
   635 		}
   636 
   637 	RemoveAllKeyRects();
   638 	while(iWinGcList)
   639 		iWinGcList->Deactivate();
   640 //
   641 	iFlags|=EFlagInvisible;		// First make it invisble
   642 	if (iParent)				// In case window wasn't fully constructed
   643 		ResetHiddenFlags();
   644 //
   645 	CWsWindow::Shutdown();
   646 	DeleteBaseArea();
   647 	CWsPointerBuffer::Disconnect(this);
   648 	iFlags&=~EFlagShutDownInProgress;
   649 	}
   650 
   651 void CWsClientWindow::Activate()
   652 	{
   653 	if (iFlags&EFlagActive)
   654 		OwnerPanic(EWservPanicWindowActive);
   655 	iFlags|=EFlagActive;
   656 
   657 	ResetHiddenFlags();
   658 
   659 	MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
   660 	if (windowTreeObserver)
   661 		{
   662 		windowTreeObserver->NodeExtentChanged(*this, FullRect());
   663 		windowTreeObserver->NodeActivated(*this);
   664 		}
   665 	}
   666 
   667 TBool CWsClientWindow::IsActivated() const
   668 	{
   669 	return (iFlags&EFlagActive)!=EFalse;
   670 	}
   671 
   672 void CWsClientWindow::SetCornerTypeL(TCornerType aCornerType, TInt aCornerFlags, TRegion *aNewBaseArea, TBool aNotifyShapeChanged)
   673 	{
   674 	TRegion *baseArea=NULL;
   675 	if (aCornerFlags&ECornerTypeMask)
   676 		OwnerPanic(EWservPanicCornerParams);
   677 	
   678  	switch (aCornerType)
   679 		{
   680 		case EWindowCornerSquare:
   681 			baseArea=new(ELeave) TRegionFix<1>();
   682 			break;
   683 		case EWindowCorner1:
   684 			baseArea=new(ELeave) TRegionFix<3>();
   685 			break;
   686 		case EWindowCorner2:
   687 		case EWindowCorner3:
   688 			baseArea=new(ELeave) TRegionFix<5>();
   689 			break;
   690 		case EWindowCorner5:
   691 			baseArea=new(ELeave) TRegionFix<9>();
   692 			break;
   693 		case EWindowCornerRegion:
   694 			User::LeaveIfNull(baseArea=aNewBaseArea);
   695 			baseArea->Offset(Origin());
   696 			break;
   697 		default:
   698 			OwnerPanic(EWservPanicCornerParams);
   699 		}
   700 	DeleteBaseArea();
   701 	iCornerData=aCornerType;
   702 	iCornerData|=aCornerFlags;
   703 	iBaseArea=baseArea;
   704 	CalcBaseArea();
   705 	ScheduleRegionUpdate(NULL);
   706 
   707 	if ( aNotifyShapeChanged )
   708 		{
   709 		MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
   710 		if (windowTreeObserver)
   711 			{
   712 			windowTreeObserver->AttributeChanged(*this, MWsWindowTreeObserver::EWindowShape);
   713 			}
   714 		}
   715 	}
   716 
   717 void CWsClientWindow::SetVisible(TBool aState)
   718 	{
   719 	if (aState)
   720 		{
   721 		if (iParent==NULL)
   722 			OwnerPanic(EWservPanicParentDeleted);
   723 		if (!(iFlags&EFlagInvisible))	// Already visible
   724 			return;
   725 		iFlags&=~EFlagInvisible;
   726 		ResetHiddenFlags();
   727 		}
   728 	else
   729 		{
   730 		if (iFlags&EFlagInvisible || !iParent)	// Already invisible or parent has been deleted
   731 			return;
   732 		TWalkWindowTreePurgeEvents wwt;
   733 		WalkWindowTree(wwt,EWalkChildren);		// Destroy all events on this and all children
   734 		iFlags|=EFlagInvisible;
   735 		ResetHiddenFlags();
   736 		}
   737 	
   738 	MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
   739 	if (windowTreeObserver)
   740 		{
   741 		windowTreeObserver->FlagChanged(*this, MWsWindowTreeObserver::EVisible, aState);
   742 		}
   743 	}
   744 
   745 void CWsClientWindow::CommandL(TInt aOpcode, const TAny *aCmdData)
   746 	{
   747 #ifdef _DEBUG
   748 	// Save root window for performing CheckTree at the end of this func.
   749 	// When aOpcode is EWsWinOpFree, this object would've been destroyed
   750 	// and a call to RootWindow() in that case would be impossible
   751 	CWsRootWindow* rootWindow=RootWindow();
   752 #endif
   753 	TWsWinCmdUnion pData;
   754 	pData.any=aCmdData;
   755 	if (CWsWindowBase::CommandL(aOpcode,pData)==EFalse)
   756 		{
   757 		switch(aOpcode)
   758 			{
   759 			case EWsWinOpActivate:
   760 				Activate();
   761 				break;
   762 			case EWsWinOpSetPos:
   763 				SetExtentL(pData.pos,NULL);
   764 				break;
   765 			case EWsWinOpSetExtent:
   766 			case EWsWinOpSetExtentErr:
   767 				SetExtentL(&pData.SetEx->pos,&pData.SetEx->size);
   768 				break;
   769 			case EWsWinOpSetSize:
   770 			case EWsWinOpSetSizeErr:
   771 				SetExtentL(NULL,pData.size);
   772 				break;
   773 			case EWsWinOpInquireOffset:
   774 				CWsClient::ReplyPoint(InquireOffset(*pData.UInt));
   775 				break;
   776 			case EWsWinOpPosition:
   777 				CWsClient::ReplyPoint(iRel.iTl);
   778 				break; 
   779 			case EWsWinOpAbsPosition:
   780 				CWsClient::ReplyPoint(iOrigin);
   781 				break;
   782 			case EWsWinOpSize:
   783 				CWsClient::ReplySize(iRel.Size());
   784 				break;
   785 			case EWsWinOpTestInvariant:
   786 				SetReply(EFalse);
   787 				break;
   788 			case EWsWinOpPointerFilter:
   789 				{
   790 				TUint old=iPointerFilter;
   791 				iPointerFilter&=~pData.PointerFilter->mask;
   792 				iPointerFilter|=pData.PointerFilter->mask&pData.PointerFilter->flags;
   793 				if (old&EPointerFilterEnterExit)
   794 					TWsPointer::ReLogWindow(this);
   795 				}
   796 				break;
   797 			case EWsWinOpSetPointerGrab:
   798 				if (*pData.Bool==EFalse)
   799 					iFlags&=~EFlagPointerGrab;
   800 				else
   801 					iFlags|=EFlagPointerGrab;
   802 				break;
   803 			case EWsWinOpClaimPointerGrab:
   804 				{
   805 				if (!iParent)
   806 					OwnerPanic(EWservPanicParentDeleted);		
   807 				
   808 				TInt errNo = TWsPointer::ClaimGrab(this,*pData.GrabControl);
   809 				if(TWsWinCmdGrabControl::ESendReply & pData.GrabControl->flags)
   810 					{
   811 					// To avoid the reply-generated-flush, only do this for the new APIs, not the old ones.					
   812 					SetReply(errNo);
   813 					}
   814 				}
   815 				break;
   816 			case EWsWinOpSetPointerCapture:
   817 				iFlags&=~(EFlagPointerCaptured|EFlagPointerCaptureDragDrop|EFlagPointerCaptureAllGroups);
   818 				if ((*pData.UInt)&RWindowBase::TCaptureFlagEnabled)
   819 					{
   820 					iFlags|=EFlagPointerCaptured;
   821 					if ((*pData.UInt)&RWindowBase::TCaptureFlagDragDrop)
   822 						iFlags|=EFlagPointerCaptureDragDrop;
   823 					if ((*pData.UInt)&RWindowBase::TCaptureFlagAllGroups)
   824 						iFlags|=EFlagPointerCaptureAllGroups;
   825 					
   826 					}
   827 				TWsPointer::ReLogPointersCurrentWindows();
   828 				break;
   829 			case EWsWinOpSetPointerCapturePriority:
   830 				iPointerCapturePriority=*pData.Int;
   831 				break;
   832 			case EWsWinOpGetPointerCapturePriority:
   833 				SetReply(iPointerCapturePriority);
   834 				break;
   835 			case EWsWinOpSetVisible:
   836 				SetVisible(*pData.Bool);
   837 				break;
   838 			case EWsWinOpScroll:
   839 				{
   840 				TPoint origin(0,0);
   841 				TRect src(TRect(origin,iRel.Size()));
   842 				src.Move(-pData.ScrollRect->offset);
   843 				Scroll(TRect(origin,iRel.Size()),pData.ScrollRect->offset,src);
   844 				}
   845 				break;
   846 			case EWsWinOpScrollClip:
   847 				{
   848 				TPoint origin(0,0);
   849 				TRect src(TRect(origin,iRel.Size()));
   850 				src.Move(-pData.ScrollRect->offset);
   851 				TRect clip(pData.ScrollRect->clip);
   852 				Scroll(clip,pData.ScrollRect->offset,src);
   853 				}
   854 				break;
   855 			case EWsWinOpScrollRect:
   856 				{
   857 				TRect src(pData.ScrollRect->rect);
   858 				Scroll(TRect(TPoint(0,0),iRel.Size()),pData.ScrollRect->offset,src);
   859 				}
   860 				break;
   861 			case EWsWinOpScrollClipRect:
   862 				{
   863 				TRect src(pData.ScrollRect->rect);
   864 				TRect clip(pData.ScrollRect->clip);
   865 				Scroll(clip, pData.ScrollRect->offset,src);
   866 				}
   867 				break;
   868 			case EWsWinOpSetOrdinalPositionPri:
   869 				iOrdinalPriority=pData.OrdinalPos->ordinalPriority;
   870 				SetOrdinalPosition(pData.OrdinalPos->pos);
   871 				break;
   872 			case EWsWinOpSetShadowHeight:
   873 				OwnerPanic(EWservPanicOpcode); //this op code is deprecated, should never be generated by the client
   874 				break;
   875 			case EWsWinOpShadowDisabled:
   876 				OwnerPanic(EWservPanicOpcode); //this op code is deprecated, should never be generated by the client
   877 				break;
   878 			case EWsWinOpRequiredDisplayMode:
   879 				if (Backup()!=NULL)
   880 					OwnerPanic(EWservPanicBackupDisplayMode);
   881 				SetReply(SetRequiredDisplayModeL(*pData.DisplayMode));
   882 				break;
   883 			case EWsWinOpGetDisplayMode:
   884 				SetReply(DisplayMode());
   885 				break;
   886 			case EWsWinOpRequestPointerRepeatEvent:
   887 				{
   888 				if (!iParent)
   889 					OwnerPanic(EWservPanicParentDeleted);
   890 				TInt errNo = TWsPointer::RequestPointerRepeatEvent(this,*pData.RequestPointerRepeatEvent);				
   891 				if(TWsWinCmdRequestPointerRepeatEvent::ERepeatFlagsSendReply & pData.RequestPointerRepeatEvent->repeatFlags)
   892 					{
   893 					SetReply(errNo);
   894 					}
   895 				}
   896 				break;
   897 			case EWsWinOpCancelPointerRepeatEventRequest:
   898 				{
   899 				TInt errNo = TWsPointer::CancelPointerRepeatEventRequest(*pData.CancelPointerRepeatEventRequest);				
   900 				if(TWsWinCmdCancelPointerRepeatEventRequest::ECancelRepeatFlagsSendReply & pData.RequestPointerRepeatEvent->repeatFlags)
   901 					{
   902 					SetReply(errNo);
   903 					}
   904 				}
   905 				break;
   906 			case EWsWinOpAllocPointerMoveBuffer:
   907 				CWsPointerBuffer::ConnectL(this,pData.AllocPointerMoveBuffer->maxNumPoints,pData.AllocPointerMoveBuffer->flags);
   908 				iFlags|=EFlagUsingPointerBuffer|EFlagHasPointerBuffer;
   909 				break;
   910 			case EWsWinOpFreePointerMoveBuffer:
   911 				CWsPointerBuffer::Disconnect(this);
   912 				iFlags&=~(EFlagUsingPointerBuffer|EFlagHasPointerBuffer);
   913 				break;
   914 			case EWsWinOpRetrievePointerMoveBuffer:
   915 				CWsPointerBuffer::RetrievePointerMoveBuffer(this,*pData.Int);
   916 				break;
   917 			case EWsWinOpEnablePointerMoveBuffer:
   918 				if (!(iFlags&EFlagHasPointerBuffer))
   919 					OwnerPanic(EWservPanicNoPointerBuffer);
   920 				iFlags|=EFlagUsingPointerBuffer;
   921 				break;
   922 			case EWsWinOpDisablePointerMoveBuffer: 
   923 				iFlags&=~EFlagUsingPointerBuffer; 
   924 				/*Fall Through*/
   925 			case EWsWinOpDiscardPointerMoveBuffer:	
   926 				CWsPointerBuffer::DiscardPointerMoveBuffer(this);
   927 				break;
   928 			case EWsWinOpAddKeyRect:
   929 				AddKeyRectL(pData.AddKeyRect->rect, pData.AddKeyRect->scanCode, pData.AddKeyRect->activatedByPointerSwitchOn);
   930 				break;
   931 			case EWsWinOpRemoveAllKeyRects:
   932 				RemoveAllKeyRects();
   933 				break;
   934 			case EWsWinOpPasswordWindow:
   935 				if (!iParent)
   936 					OwnerPanic(EWservPanicParentDeleted);
   937 				CWsPassword::SetPasswordWindowL(this, *pData.PasswordMode);
   938 				break;
   939 			case EWsWinOpEnableBackup:
   940 				if (!iParent)
   941 					OwnerPanic(EWservPanicParentDeleted);
   942 				if (*pData.UInt==0)
   943 					iBackupsRequested|=EWindowBackupAreaBehind;		//For backwards compatibility
   944 				else
   945 					iBackupsRequested|=*pData.UInt;
   946 				break;
   947 			case EWsWinOpFadeBehind:
   948 				{
   949 				if (!iParent)
   950 					OwnerPanic(EWservPanicParentDeleted);
   951 				
   952 				TUint8 blackMap;
   953 				TUint8 whiteMap;
   954 				iScreen->GetFadingParams(blackMap,whiteMap);
   955 				SetFadeBehind(*pData.Bool);
   956 				TWalkWindowTreeSetFaded wwt(*pData.Bool,this,blackMap,whiteMap);
   957 				WalkWindowTree(wwt,EWalkBehind);
   958 				}
   959 				break;
   960 			case EWsWinOpGetIsFaded:
   961 				SetReply(iFadeCount);
   962 				break;
   963 			case EWsWinOpGetIsNonFading:
   964 				SetReply(iFlags&EFlagNonFadingWindow);
   965 				break;
   966 			case EWsWinOpMoveToGroup:
   967 				if (!iParent)
   968 					OwnerPanic(EWservPanicParentDeleted);
   969 				if (iParent->WinType()!=EWinTypeGroup)
   970 					OwnerPanic(EWservPanicNotTopClient);
   971 				((CWsTopClientWindow*)this)->DoMoveWindowL(*pData.Int);
   972 				break;
   973 			case EWsWinOpTestLowPriorityRedraw:
   974 				{
   975 				// This is purely for testing purposes
   976 				// Returns the redraw priority
   977 				TUint priority=0;
   978 				TPckgBuf<TUint> priBuf;
   979 				priority=WsOwner()->RedrawQueue()->RedrawPriority((CWsWindowRedraw*)this->iRedraw);
   980 				priBuf()=priority;
   981 				CWsClient::ReplyBuf(priBuf);
   982 				}
   983 				break;
   984 			case EWsWinOpEnableVisibilityChangeEvents:
   985 				iFlags |= EFlagGeneratesVisibilityEvents;
   986 				SetupVisibleRegionTracking(ETrue);
   987 				if (iFlags&EFlagActive)
   988 					{
   989 					iScreen->DoRedrawNow();
   990 					PossibleVisibilityChangedEvent(ETrue);
   991 					}
   992 				break;
   993 			case EWsWinOpDisableVisibilityChangeEvents:
   994 				iFlags &= ~EFlagGeneratesVisibilityEvents;
   995 				SetupVisibleRegionTracking(EFalse);
   996 				break;
   997 			case EWsWinOpSetTransparentRegion:
   998 				{
   999 				if (IsTranslucent())
  1000 					{
  1001 					TInt recs=*pData.Int;
  1002 	 				RWsRegion* reg=recs>0? GetRegionFromClientL(iWsOwner,recs) : new(ELeave) RWsRegion;
  1003 	 				SetUserTransparentRegion(reg);
  1004 					SetReply(KErrNone);
  1005 					}
  1006 				else
  1007 					{
  1008 					OwnerPanic(EWservPanicTransparencyObjNotCreated);	
  1009 					}				
  1010 				}
  1011 				break;
  1012 			case EWsWinOpSetTransparencyPolicy:
  1013 				{
  1014 				if (IsTranslucent())
  1015 					SetReply(KErrNone);
  1016 				else
  1017 					OwnerPanic(EWservPanicTransparencyObjNotCreated);
  1018 				}
  1019 				break;
  1020 			case EWsWinOpSetTransparencyAlphaChannel:
  1021 				{
  1022 				iFlags |= static_cast<TUint>(EFlagHasAlpha);
  1023 				SetReply(KErrNone);
  1024 				
  1025 				MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
  1026 				if (windowTreeObserver)
  1027 					{
  1028 					windowTreeObserver->FlagChanged(*this, MWsWindowTreeObserver::EAlphaChannelTransparencyEnabled, ETrue);
  1029 					}
  1030 				break;
  1031 				}			
  1032 			default:
  1033 				if (iRedraw->CommandL(aOpcode,pData)==EFalse)
  1034 					{
  1035 					OwnerPanic(EWservPanicOpcode);
  1036 					}
  1037 			}
  1038 		}
  1039 #if defined(_DEBUG)
  1040 	rootWindow->CheckTree();
  1041 #endif
  1042 	}
  1043 
  1044 void CWsClientWindow::GcActivated(CWsGc *aGc)
  1045 	{
  1046 	aGc->SetNextWinGc(iWinGcList);
  1047 	iWinGcList=aGc;
  1048 	}
  1049 
  1050 void CWsClientWindow::GcDeactivated(CWsGc *aGc)
  1051 	{
  1052 	if (aGc==iWinGcList)
  1053 		iWinGcList=aGc->NextWinGc();
  1054 	else
  1055 		{
  1056 		CWsGc *gc=iWinGcList;
  1057 		CWsGc *next;
  1058 		FOREVER
  1059 			{
  1060 			next=gc->NextWinGc();
  1061 			WS_ASSERT_DEBUG(next!=NULL, EWsPanicBadActiveGcList);
  1062 			if (next==aGc)
  1063 				{
  1064 				gc->SetNextWinGc(next->NextWinGc());
  1065 				break;
  1066 				}
  1067 			gc=next;
  1068 			}
  1069 		}
  1070 	aGc->SetNextWinGc(NULL);
  1071 	}
  1072 
  1073 void CWsClientWindow::ReactivateGcs()
  1074 	{
  1075 	for (CWsGc * gc = iWinGcList; gc; gc = gc->NextWinGc())
  1076 		{
  1077 		gc->Reactivate();
  1078 		}
  1079 	}
  1080 
  1081 void CWsClientWindow::OffsetUserTransparentRegion(const TPoint& aOffset)
  1082 	{
  1083 	if (iUserDefinedTransparentRegion)
  1084 		{
  1085 		iUserDefinedTransparentRegion->Offset(aOffset);
  1086 		}	
  1087 	}
  1088 
  1089 void CWsClientWindow::SetUserTransparentRegion(RWsRegion* aRegion)
  1090 	{
  1091 	if (iUserDefinedTransparentRegion)
  1092 		{
  1093 		iUserDefinedTransparentRegion->Close();
  1094 		delete iUserDefinedTransparentRegion;
  1095 		iUserDefinedTransparentRegion = 0;
  1096 		}
  1097 		
  1098 	if (aRegion)
  1099 		{		
  1100 		aRegion->Offset(iOrigin);
  1101 		iUserDefinedTransparentRegion=aRegion;
  1102 		}
  1103 		
  1104 	SetUserOpaqueRegion();
  1105 	}
  1106 
  1107 void CWsClientWindow::SetUserOpaqueRegion()
  1108 	{
  1109 	if (iUserDefinedOpaqueRegion)
  1110 		{
  1111 		iUserDefinedOpaqueRegion->Close();
  1112 		delete iUserDefinedOpaqueRegion;
  1113 		iUserDefinedOpaqueRegion = 0;
  1114 		}
  1115 	if (iUserDefinedTransparentRegion)
  1116 		{
  1117 		iUserDefinedOpaqueRegion=new RWsRegion;
  1118 		if (iUserDefinedOpaqueRegion)
  1119 			{
  1120 			iUserDefinedOpaqueRegion->Copy(*iBaseArea);
  1121 			iUserDefinedOpaqueRegion->SubRegion(*iUserDefinedTransparentRegion);
  1122 			if (iUserDefinedOpaqueRegion->CheckError() || iUserDefinedOpaqueRegion->Count() == 0)
  1123 				{
  1124 				iUserDefinedOpaqueRegion->Close();
  1125 				delete iUserDefinedOpaqueRegion;
  1126 				iUserDefinedOpaqueRegion = 0;
  1127 				}
  1128 			}
  1129 		// Intentionally not sending this notification during destruction (when SetUserTransparentRegion is called from d'tor)
  1130 		MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
  1131 		if (windowTreeObserver)
  1132 			{
  1133 			windowTreeObserver->TransparentRegionChanged(*this, *iUserDefinedTransparentRegion, iUserDefinedOpaqueRegion);
  1134 			}
  1135 		}
  1136 	}
  1137 
  1138 /** Checks whether this window is in front of aWin.
  1139 
  1140 @param aWin A window.
  1141 @return EFalse if aWin is the same or is in front of this, ETrue otherwise.
  1142 @internalComponent
  1143 * released
  1144 */	
  1145 TBool CWsClientWindow::IsInfrontOf(const CWsWindowBase* aWin) const
  1146 	{
  1147 	TInt thisDepth=Depth();
  1148 	TInt otherDepth=aWin->Depth();
  1149 	const CWsWindowBase *thisWin=this;
  1150 	const CWsWindowBase *otherWin=aWin;
  1151 	if (thisDepth>otherDepth)
  1152 		{
  1153 		for (TInt count=thisDepth-otherDepth;count>0;count--)
  1154 			thisWin=thisWin->BaseParent();
  1155 		}
  1156 	else
  1157 		{
  1158 		for (TInt count=otherDepth-thisDepth;count>0;count--)
  1159 			otherWin=otherWin->BaseParent();
  1160 		}
  1161 	if (thisWin==otherWin)
  1162 		return thisDepth>otherDepth;
  1163 	while(thisWin->BaseParent()!=otherWin->BaseParent())
  1164 		{
  1165 		thisWin=thisWin->BaseParent();
  1166 		otherWin=otherWin->BaseParent();
  1167 		}
  1168 	const CWsWindowBase *win=thisWin->BaseParent()->BaseChild();
  1169 	FOREVER
  1170 		{
  1171 		if (win==otherWin)
  1172 			{
  1173 			return EFalse;
  1174 			}
  1175 		if (win==thisWin)
  1176 			return ETrue;
  1177 		win=win->NextSibling();
  1178 		}
  1179 	}
  1180 	
  1181 CWsTopClientWindow* CWsClientWindow::TopClientWindow()
  1182 	{
  1183 	if (iParent==NULL)
  1184 		OwnerPanic(EWservPanicParentDeleted);
  1185 	CWsWindowBase* win=this;
  1186 	while(win->BaseParent()->WinType()!=EWinTypeGroup)
  1187 		win=win->BaseParent();
  1188 	return static_cast<CWsTopClientWindow*>(win);
  1189 	}
  1190 
  1191 const TRegion &CWsClientWindow::InvalidArea() const
  1192 	{
  1193 	return(iRedraw->InvalidArea());
  1194 	}
  1195 
  1196 
  1197 TUint CWsClientWindow::RedrawPriority(TInt *aShift) const
  1198 	{
  1199 	TUint ordinalPos=OrdinalPosition(EFalse)+1;
  1200 	if (ordinalPos>15)	// Algorithm only works upto 15 , make all windows after 15 equal in priority
  1201 		ordinalPos=15;
  1202 	TInt shift;
  1203 	TUint parent=((CWsClientWindow *)iParent)->RedrawPriority(&shift);
  1204 	if (shift>0)
  1205 		shift--;
  1206 	if (aShift)
  1207 		*aShift=shift;
  1208 	return(parent+(ordinalPos<<(shift*KWinRedrawPriBitsPerLevel)));
  1209 	}
  1210 
  1211 TDblQue<TPointerKeyList> *CWsClientWindow::PointerKeyList() const
  1212 	{
  1213 	return(iPointerKeyList);
  1214 	}
  1215 
  1216 void CWsClientWindow::AddKeyRectL(const TRect &aRect, TInt aScanCode, TBool aActivatedByPointerSwitchOn)
  1217 	{
  1218 	if (!iPointerKeyList)
  1219 		iPointerKeyList=new(ELeave) TDblQue<TPointerKeyList>(_FOFF(TPointerKeyList,iQue));
  1220 	TPointerKeyList *pkl=new(ELeave) TPointerKeyList();
  1221 	iPointerKeyList->AddLast(*pkl);
  1222 	pkl->iRect=aRect;
  1223 	pkl->iScanCode=aScanCode;
  1224 	pkl->iActivatedByPointerSwitchOn=aActivatedByPointerSwitchOn;
  1225 	}
  1226 
  1227 void CWsClientWindow::RemoveAllKeyRects()
  1228 	{
  1229 	if (iPointerKeyList)
  1230 		{
  1231 		TPointerKeyList *pkl=NULL;
  1232 		for(TDblQueIter<TPointerKeyList> iter(*iPointerKeyList);(pkl=iter++)!=NULL;)
  1233 			{
  1234 			pkl->iQue.Deque();
  1235 			delete pkl;
  1236 			}
  1237 		delete iPointerKeyList;
  1238 		iPointerKeyList=NULL;
  1239 		}
  1240 	}
  1241 
  1242 TBool CWsClientWindow::IsHidden()
  1243 	{
  1244 	return (!IsVisible()) || VisibleRegion().IsEmpty();
  1245 	}
  1246 
  1247 void CWsClientWindow::SetFaded(TBool aFade, TUint8 aBlackMap, TUint8 aWhiteMap, TBool aNotifyObserver)
  1248 	{
  1249 	TBool stateChanged;
  1250 	SetFaded(aFade, aBlackMap, aWhiteMap, aNotifyObserver, stateChanged);
  1251 	}
  1252 
  1253 void CWsClientWindow::SetFaded(TBool aFade, TUint8 aBlackMap, TUint8 aWhiteMap, TBool aNotifyObserver, TBool& aStateChanged)
  1254 	{
  1255 	iBlackMap=aBlackMap;
  1256 	iWhiteMap=aWhiteMap;
  1257 	const TBool mapAltered = (iBlackMap != aBlackMap) || (iWhiteMap != aWhiteMap); 
  1258 	const TInt oldFadeCount = iFadeCount;
  1259 
  1260 	if (iAbsoluteFading) 
  1261 		{ 
  1262 		if (aFade) 
  1263 			{ 
  1264 			iFadeCount = 1; 
  1265 			} 
  1266 		else 
  1267 			{ 
  1268 			iFadeCount = 0; 
  1269 			} 
  1270 		} 
  1271 	else 
  1272 		{ 
  1273 		if (aFade) 
  1274 			{ 
  1275 			++iFadeCount; 
  1276 			} 
  1277 		else if (iFadeCount > 0) 
  1278 			{ 
  1279 			--iFadeCount; 
  1280 			} 
  1281 		}
  1282 	
  1283 	//Fade state only changes when iFadeCount transitions from 0 to 1 or from 1 to 0.
  1284  	aStateChanged = (iFadeCount==0 || oldFadeCount==0) && (oldFadeCount != iFadeCount);
  1285  	if (!Screen()->ChangeTracking() && CWsTop::IsFadeEnabled() && (aStateChanged || mapAltered) ) 
  1286  		{
  1287  		Screen()->AcceptFadeRequest(this, (iFadeCount > 0));
  1288  		}
  1289 
  1290 	const TBool doNotify = iAbsoluteFading ? aStateChanged : (oldFadeCount != iFadeCount);
  1291 	MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
  1292 	if (windowTreeObserver && aNotifyObserver && doNotify)
  1293 		{
  1294 		windowTreeObserver->FadeCountChanged(*this, iFadeCount);
  1295 		}	
  1296 	}
  1297 
  1298 void CWsClientWindow::ResetHiddenFlagsInParentAndChildren()
  1299 	{
  1300 	ResetHiddenFlag();
  1301 	SetElementOpacity(IsVisible() ? 0xFF : 0x00);	// Update element visibility
  1302 	for(CWsClientWindow* child=Child();child;child=child->NextSibling())
  1303 		{
  1304 		child->ResetHiddenFlagsInParentAndChildren();
  1305 		}
  1306 	}
  1307 
  1308 const TRegion& CWsClientWindow::WindowArea() const
  1309 	{
  1310 	WS_ASSERT_DEBUG(iBaseArea, EWsPanicRegionNull);
  1311 	return *iBaseArea;
  1312 	}
  1313 
  1314 void CWsClientWindow::Invalidate(const TRect * aRect)
  1315 	{
  1316 	iRedraw->Invalidate(aRect);
  1317 	}
  1318 
  1319 void CWsClientWindow::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
  1320 	{
  1321 	if (IsVisible())
  1322 		{
  1323 		iScreen->ScheduleRegionUpdate(aDefinitelyDirty);
  1324 		}
  1325 	}
  1326 
  1327 void CWsClientWindow::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
  1328 	{
  1329 	if (!IsHidden())
  1330 		{
  1331 		iScreen->AddRedrawRegion(aRegion, aSchedule, aDepth);
  1332 		}
  1333 	}
  1334 
  1335 void CWsClientWindow::SendState(MWsWindowTreeObserver& aWindowTreeObserver) const
  1336 	{	
  1337 	CWsWindow::SendState(aWindowTreeObserver);
  1338 	
  1339 	if(iFadeCount > 0)
  1340 		{
  1341 		aWindowTreeObserver.FadeCountChanged(*this, iFadeCount);
  1342 		}
  1343 	
  1344 	if(iUserDefinedTransparentRegion)
  1345 		{
  1346 		aWindowTreeObserver.TransparentRegionChanged(*this, *iUserDefinedTransparentRegion, iUserDefinedOpaqueRegion);
  1347 		}
  1348 
  1349 	if(HasElement())
  1350 		{
  1351 		CWindowElementSet& windowElementSet = Screen()->WindowElements();
  1352 		const TBackgroundAttributes		*bElementAttr;
  1353 		const RArray<TPlacedAttributes>	*pElementsAttr;
  1354 		
  1355 		TInt ret = windowElementSet.FindElements(*this, bElementAttr, pElementsAttr);
  1356 		if(ret == KErrNone)
  1357 			{
  1358 			MWsElement* element = bElementAttr->iElement;
  1359 			if (element)
  1360 				aWindowTreeObserver.ElementAdded(*this, *element);
  1361 			}
  1362 		}
  1363 	
  1364 	}
  1365 
  1366 TBool CWsClientWindow::IsDSAHost() const
  1367 	{
  1368 	TBool res = CWsWindow::IsDSAHost();
  1369 	if ( !res )
  1370 		{ // check for grace period when DSA is being restarted (after aborting but before client started DSA again)
  1371 		res = Screen()->IsDSAClientWindow( this );
  1372 		}
  1373 	return res;
  1374 	}
  1375 
  1376 void CWsClientWindow::UpdateElementExtent(const TPoint* aOffset)
  1377 	{
  1378 	if (Redraw()->HasElement())
  1379 		{
  1380 		Screen()->WindowElements().UpdateElementExtent(*this, aOffset);
  1381 		}
  1382 	}
  1383 
  1384 void CWsClientWindow::SetElementOpacity(TInt aOpacity)
  1385 	{
  1386 	if (Redraw()->HasElement())
  1387 		{
  1388 		Screen()->WindowElements().SetElementOpacity(*this,aOpacity);
  1389 
  1390 		} 
  1391 	}
  1392 
  1393 TRect CWsClientWindow::GetOriginalSrcElementRect() const
  1394     {
  1395     return iOriginalSrcElementRect;
  1396     }
  1397 TRect CWsClientWindow::GetOriginalDestElementRect() const
  1398     {
  1399     return iOriginalDestElementRect;
  1400     }
  1401     
  1402 //
  1403 // Code for CWsTopClientWindow, a client window that connects to a group window //
  1404 //
  1405 
  1406 CWsTopClientWindow::CWsTopClientWindow(CWsClient* aOwner, CScreen* aScreen) : CWsClientWindow(aOwner, aScreen)
  1407 	{
  1408 	}
  1409 
  1410 void CWsTopClientWindow::ConstructL(const TWsClCmdCreateWindow &cmd, CWsWindowBase *aParent, TBool aScreenDeviceIsInvalid)
  1411 	{
  1412 	iFlags|=EFlagIsTopClientWindow;
  1413 	CWsClientWindow::ConstructL(cmd, aParent, aScreenDeviceIsInvalid);
  1414 	}
  1415 
  1416 void CWsTopClientWindow::SetInactive()
  1417 	{
  1418 	iFlags&=~EFlagActive;
  1419 	ResetHiddenFlags();
  1420 	}
  1421 
  1422 void CWsTopClientWindow::SetScreenDeviceValidState(TBool aState)
  1423 	{
  1424 	if (SetScreenDeviceValidStateFlag(aState))
  1425 		ResetHiddenFlags();
  1426 	}
  1427 
  1428 TBool CWsTopClientWindow::SetScreenDeviceValidStateFlag(TBool aState)
  1429 	{
  1430 	TBool isSet=iFlags&EFlagScreenDeviceInvalid;
  1431 	if (!isSet==!aState)
  1432 		{
  1433 		if (aState)
  1434 			iFlags&=~EFlagScreenDeviceInvalid;
  1435 		else
  1436 			iFlags|=EFlagScreenDeviceInvalid;
  1437 		
  1438 		MWsWindowTreeObserver* windowTreeObserver = iScreen->WindowTreeObserver();
  1439 		if (windowTreeObserver)
  1440 			{
  1441 			windowTreeObserver->FlagChanged(*this, MWsWindowTreeObserver::EScreenDeviceValid, aState);
  1442 			}
  1443 		
  1444 		return ETrue;
  1445 		}
  1446 	return EFalse;
  1447 	}
  1448 
  1449 void CWsTopClientWindow::SetOrdinalPosition(TInt aPos)
  1450 	{
  1451 	if (!iParent)
  1452 		{
  1453 		OwnerPanic(EWservPanicParentDeleted);
  1454 		}
  1455 	if (CheckOrdinalPositionChange(aPos))
  1456 		{
  1457 		CWsWindowBase::SetOrdinalPosition(aPos);
  1458 		CWsTop::TriggerRedraws(RootWindow());
  1459 		}
  1460 	}
  1461 
  1462 void CWsTopClientWindow::DoMoveWindowL(TInt aIdentifier)
  1463 	{
  1464 	CWsWindowGroup* group=CWsWindowGroup::WindowGroupFromIdentifierL(aIdentifier);
  1465 	if (group==iParent)
  1466 		return;
  1467 	if (group->WsOwner()!=WsOwner())
  1468 		User::Leave(KErrNotFound);
  1469 	ChangeWindowPosition(0, group);
  1470 	CWsTop::TriggerRedraws(RootWindow());
  1471 	}
  1472 
  1473 TUint CWsTopClientWindow::RedrawPriority(TInt *aShift) const
  1474 	{
  1475 	TUint ordinalPos=OrdinalPosition(EFalse);
  1476 	if (ordinalPos>KWinRedrawPriMaxOrdinal)	// Algorithm only works for upto KWinRedrawPriMaxOrdinal windows,
  1477 		ordinalPos=KWinRedrawPriMaxOrdinal;	// make all windows after this equal in priority
  1478 	if (aShift)
  1479 		*aShift=KWinRedrawPriMaxLevel;
  1480 	return(ordinalPos<<(KWinRedrawPriMaxLevel*KWinRedrawPriBitsPerLevel));
  1481 	}
  1482