os/graphics/windowing/windowserver/nga/SERVER/openwfc/windowelementset.cpp
changeset 0 bde4ae8d615e
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/os/graphics/windowing/windowserver/nga/SERVER/openwfc/windowelementset.cpp	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,2004 @@
     1.4 +// Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
     1.5 +// All rights reserved.
     1.6 +// This component and the accompanying materials are made available
     1.7 +// under the terms of "Eclipse Public License v1.0"
     1.8 +// which accompanies this distribution, and is available
     1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
    1.10 +//
    1.11 +// Initial Contributors:
    1.12 +// Nokia Corporation - initial contribution.
    1.13 +//
    1.14 +// Contributors:
    1.15 +//
    1.16 +// Description:
    1.17 +//
    1.18 +
    1.19 +#include <e32std.h>
    1.20 +#include "cliwin.h"
    1.21 +#include "server.h"
    1.22 +#include "windowelementset.h"
    1.23 +#include "regionextend.h"
    1.24 +#include <graphics/wsscene.h>
    1.25 +#include <graphics/wsgraphicdrawer.h>
    1.26 +#include "panics.h"
    1.27 +#include "wstop.h"
    1.28 +
    1.29 +#include "EVENT.H"
    1.30 +
    1.31 +#ifndef KCacheListGranularity
    1.32 +	enum	{KCacheListGranularity = 8};
    1.33 +#endif
    1.34 +#ifndef KCacheExtensionReserve
    1.35 +	enum	{KCacheExtensionReserve = 1};
    1.36 +#endif
    1.37 +	
    1.38 +
    1.39 +
    1.40 +/** Element to gc rotation function
    1.41 +*/
    1.42 +LOCAL_C inline CFbsBitGc::TGraphicsOrientation ElementToGcRotation(MWsElement::TElementRotation aElementRotation)
    1.43 +	{
    1.44 +	CFbsBitGc::TGraphicsOrientation gcRotation = CFbsBitGc::EGraphicsOrientationNormal;
    1.45 +
    1.46 +	switch (aElementRotation)
    1.47 +		{
    1.48 +		case MWsElement::EElementAntiClockwise90:
    1.49 +			gcRotation = CFbsBitGc::EGraphicsOrientationRotated90;
    1.50 +			break;
    1.51 +		case MWsElement::EElementAntiClockwise180:
    1.52 +			gcRotation = CFbsBitGc::EGraphicsOrientationRotated180;
    1.53 +			break;
    1.54 +		case MWsElement::EElementAntiClockwise270:
    1.55 +			gcRotation = CFbsBitGc::EGraphicsOrientationRotated270;
    1.56 +			break;
    1.57 +		default:
    1.58 +			break;
    1.59 +		}
    1.60 +	return gcRotation;
    1.61 +	}
    1.62 +
    1.63 +//
    1.64 +// CWindowElement
    1.65 +//
    1.66 +
    1.67 +CWindowElement::CWindowElement(const CWsClientWindow& aWindow) :
    1.68 +	iWindow(aWindow), iCache(KCacheListGranularity), iHighestReusedIndex(-1)
    1.69 +	{}
    1.70 +
    1.71 +
    1.72 +/** Finds all elements within the window that overlap with the region and 
    1.73 +marks them as unassigned.
    1.74 +If any part of these unassigned elements are outside the region, they are also 
    1.75 +marked overlapping and locked config.
    1.76 +
    1.77 +@param aRegion The region that is being redrawn.
    1.78 +@return KErrNone on success or KErrNoMemory if critical calculations could not be made.
    1.79 +
    1.80 +@see CWindowElementSet::UnassignPlacedElements
    1.81 +**/
    1.82 +TInt CWindowElement::UnassignPlacedElements(const TRegion& aRegion,TInt aGcDrawingCount)
    1.83 +	{
    1.84 +	iFlags = 0;	//reset flags
    1.85 +	iStartGcDrawingCount = aGcDrawingCount;	//Detect any drawing between calls
    1.86 +	iBackgroundElement.SetDrawnOverLast(iBackgroundElement.DrawnOver());
    1.87 +	iNonRedrawVisibleRegion.Clear();
    1.88 +	iNonRedrawVisibleRegion.Copy(const_cast<CWsClientWindow&>(iWindow).VisibleRegion());
    1.89 +	iNonRedrawVisibleRegion.SubRegion(aRegion);
    1.90 +	if (iNonRedrawVisibleRegion.CheckError())
    1.91 +		{	//memory error. Still needs to cope if whole window is being redrawn
    1.92 +		iNonRedrawVisibleRegion.Clear();
    1.93 +		TInt fullWindowRedraw = TRegionExtend::Cast(aRegion).TestDifference(const_cast<CWsClientWindow&>(iWindow).VisibleRegion());
    1.94 +		if (fullWindowRedraw != 0)
    1.95 +			{	//not a full window redraw, return immediately
    1.96 +			return KErrNoMemory;
    1.97 +			}
    1.98 +		}	
    1.99 +	TInt placedElements = iPlacedElements.Count();
   1.100 +	TInt unassignedCount = 0;
   1.101 +	TInt ii;
   1.102 +	TBool cacheError = EFalse;
   1.103 +
   1.104 +	STACK_REGION tempSurfaceBaseArea;
   1.105 +	if (iCache.Reserve(placedElements + KCacheExtensionReserve) < KErrNone)	//quite often there may need to be room for more
   1.106 +		{	//failed to reserve space
   1.107 +		cacheError = ETrue;
   1.108 +		}
   1.109 +	TRect tempExtent;
   1.110 +	TRect tempIntersect;
   1.111 +	
   1.112 +	for (ii=0; ii<placedElements; ii++)
   1.113 +		{
   1.114 +		if (!cacheError)
   1.115 +			{	//cache is ok up to this point but does not contain cache for this element yet
   1.116 +			RRegion newRegion;
   1.117 +			if (iCache.Append(newRegion) < KErrNone)
   1.118 +				{
   1.119 +				cacheError = ETrue;
   1.120 +				}
   1.121 +			newRegion.Close();
   1.122 +			}
   1.123 +
   1.124 +		iPlacedElements[ii].iElement->GetDestinationRectangle(tempExtent);
   1.125 +		TInt regionReturn = TRegionExtend::Cast(aRegion).TestDifference(tempExtent);
   1.126 +		iPlacedElements[ii].SetDrawnOverLast(iPlacedElements[ii].DrawnOver());	//allows detection of change of state
   1.127 +		if (!(regionReturn&TRegionExtend::ENoIntersect))
   1.128 +			{	//redraw region intersects with element
   1.129 +			iPlacedElements[ii].SetUnassigned();
   1.130 +			iPlacedElements[ii].SetDrawnOver(EFalse);
   1.131 +			unassignedCount++;
   1.132 +			
   1.133 +			if (!(regionReturn&TRegionExtend::EAdd))
   1.134 +				{	//surface is entirely within the redraw region
   1.135 +				iPlacedElements[ii].SetOverlapping(EFalse);
   1.136 +				iPlacedElements[ii].SetLockedConfig(EFalse);
   1.137 +				if(!cacheError)
   1.138 +					{
   1.139 +					TRect destinationRect;
   1.140 +					tempSurfaceBaseArea.Clear();
   1.141 +                    iPlacedElements[ii].iElement->GetDestinationRectangle(destinationRect);
   1.142 +					tempSurfaceBaseArea.AddRect(destinationRect);
   1.143 +					if (!tempSurfaceBaseArea.CheckError())
   1.144 +						{
   1.145 +						tempSurfaceBaseArea.Offset(tempExtent.iTl);	//cache needs to be in absolute coords
   1.146 +						iCache[ii].Copy(tempSurfaceBaseArea);
   1.147 +						}
   1.148 +					if (tempSurfaceBaseArea.CheckError() || iCache[ii].CheckError())
   1.149 +						{	//cache is no good from this point onwards
   1.150 +						iCache[ii].Close();
   1.151 +						iCache.Remove(ii);
   1.152 +						cacheError = ETrue;
   1.153 +						}
   1.154 +					}
   1.155 +				}
   1.156 +			else
   1.157 +				{
   1.158 +                TRect destinationRect;
   1.159 +                tempSurfaceBaseArea.Clear();
   1.160 +                iPlacedElements[ii].iElement->GetDestinationRectangle(destinationRect);
   1.161 +                tempSurfaceBaseArea.AddRect(destinationRect);
   1.162 +				if (tempSurfaceBaseArea.CheckError())
   1.163 +					{	//critical failure, clear cache and return error
   1.164 +					for (ii=iCache.Count(); ii>0; ii--)
   1.165 +						{
   1.166 +						iCache[ii-1].Close();
   1.167 +						iCache.Remove(ii-1);
   1.168 +						}
   1.169 +					iCache.Reset();
   1.170 +					iNonRedrawVisibleRegion.Close();
   1.171 +					tempSurfaceBaseArea.Close();
   1.172 +					return KErrNoMemory;
   1.173 +					}
   1.174 +				else
   1.175 +					{
   1.176 +					tempSurfaceBaseArea.Offset(tempExtent.iTl);	//change to absolute coordinates
   1.177 +					regionReturn = TRegionExtend::Cast(iNonRedrawVisibleRegion ).TestDifference(tempSurfaceBaseArea);
   1.178 +					if (regionReturn&TRegionExtend::ENoIntersect)
   1.179 +						{	//element base area entirely inside redraw region
   1.180 +						iPlacedElements[ii].SetOverlapping(EFalse);
   1.181 +						iPlacedElements[ii].SetLockedConfig(EFalse);
   1.182 +						}
   1.183 +					else
   1.184 +						{	//element base area is at least partly within non redraw visible region
   1.185 +						iPlacedElements[ii].SetOverlapping(ETrue);
   1.186 +						iPlacedElements[ii].SetLockedConfig(ETrue);
   1.187 +						}
   1.188 +					if(!cacheError)
   1.189 +						{
   1.190 +						iCache[ii].Copy(tempSurfaceBaseArea);
   1.191 +						if (iCache[ii].CheckError())
   1.192 +							{	//cache is no good from this point onwards
   1.193 +							iCache[ii].Close();
   1.194 +							iCache.Remove(ii);
   1.195 +							cacheError = ETrue;
   1.196 +							}
   1.197 +						}
   1.198 +					}
   1.199 +				}
   1.200 +			}
   1.201 +		else
   1.202 +			{	//element does not intersect with redraw region
   1.203 +			iPlacedElements[ii].SetUnassigned(EFalse);
   1.204 +			}
   1.205 +		}
   1.206 +	tempSurfaceBaseArea.Close();
   1.207 +	
   1.208 +	if (iRemovedSurfaces.Reserve(unassignedCount) == KErrNone)
   1.209 +		{
   1.210 +		iRemovedSurfacesValid = ETrue;
   1.211 +		}
   1.212 +	else
   1.213 +		{
   1.214 +		iRemovedSurfacesValid = EFalse;
   1.215 +		}
   1.216 +	return KErrNone;
   1.217 +	}
   1.218 +
   1.219 +/** Marks all elements that have been assigned since the unassign as drawn over.
   1.220 +
   1.221 +@see CWindowElementSet::FlagAssignedElementsDrawnOver
   1.222 +**/
   1.223 +void CWindowElement::FlagAssignedElementsDrawnOver(TInt aGcDrawingCount)
   1.224 +	{
   1.225 +	iStartGcDrawingCount = aGcDrawingCount;
   1.226 +	if (iBackgroundElement.iElement)
   1.227 +		{
   1.228 +		iBackgroundElement.SetDrawnOver();
   1.229 +		}
   1.230 +	TInt placedElementCount = iPlacedElements.Count();
   1.231 +	for (TInt ii = 0; ii < placedElementCount; ii++)
   1.232 +		{
   1.233 +		if(!iPlacedElements[ii].Unassigned() && iPlacedElements[ii].LockedConfig())
   1.234 +			{
   1.235 +			iPlacedElements[ii].SetDrawnOver();
   1.236 +			}
   1.237 +		else if(iPlacedElements[ii].Unassigned() && iPlacedElements[ii].LockedConfig() &&
   1.238 +				!iPlacedElements[ii].ChangedClip() && iPlacedElements[ii].Overlapping())
   1.239 +			{
   1.240 +			iPlacedElements[ii].SetDrawnOver();
   1.241 +			}
   1.242 +		}
   1.243 +	}
   1.244 +
   1.245 +/** Checks whether a elements configuration is the same as that of the surface configuration, taking
   1.246 +into account the stateflags (TPlacedAttributes) passed in.
   1.247 +It must match surfaceId and extent.
   1.248 +If it is a locked config it will need to match viewport and orientation as well.
   1.249 +
   1.250 +@param aElement Element to compare against.
   1.251 +@param aSurfaceConfiguration Configuration to compare.
   1.252 +@param aStateFlags The TPlacedAttributes flags for the element.
   1.253 +@return ETrue if a match, otherwise EFalse.
   1.254 +*/
   1.255 +TBool CWindowElement::IsElementAMatch(MWsElement& aElement, const TSurfaceConfiguration& aSurfaceConfiguration,
   1.256 +		TInt aStateFlags)
   1.257 +	{
   1.258 +	TBool match = EFalse;
   1.259 +	
   1.260 +	TRect tempElementExtent;
   1.261 +	TRect tempCRPExtent;
   1.262 +	aElement.GetDestinationRectangle(tempElementExtent);
   1.263 +	aSurfaceConfiguration.GetExtent(tempCRPExtent);
   1.264 +	if (tempElementExtent == tempCRPExtent)
   1.265 +		{
   1.266 +		TSurfaceId tempElementSurfaceId;
   1.267 +		TSurfaceId tempCRPSurfaceId;
   1.268 +		aSurfaceConfiguration.GetSurfaceId(tempElementSurfaceId);
   1.269 +		tempCRPSurfaceId = aElement.ConnectedSurface();
   1.270 +		if (tempElementSurfaceId == tempCRPSurfaceId)
   1.271 +			{
   1.272 +			match = ETrue;
   1.273 +			if (aStateFlags&TPlacedAttributes::ELockedConfig)
   1.274 +				{
   1.275 +				match = EFalse;
   1.276 +				TBool orientationOK = EFalse;
   1.277 +				TRect tempCRPViewport;
   1.278 +				aSurfaceConfiguration.GetViewport(tempCRPViewport);
   1.279 +				//if explicit viewport flagged
   1.280 +				TRect tempElementViewport;
   1.281 +				aElement.GetSourceRectangle(tempElementViewport);
   1.282 +				if (tempElementViewport == tempCRPViewport)
   1.283 +					{
   1.284 +					CFbsBitGc::TGraphicsOrientation tempElementOrientation;
   1.285 +					MWsElement::TElementRotation tempCRPRotation;
   1.286 +					CFbsBitGc::TGraphicsOrientation tempCRPOrientation;
   1.287 +					tempElementOrientation = aSurfaceConfiguration.Orientation();
   1.288 +					tempCRPRotation = aElement.SourceRotation();
   1.289 +					tempCRPOrientation = ElementToGcRotation(tempCRPRotation);	//convert to common type
   1.290 +					if (tempElementOrientation == tempCRPOrientation)
   1.291 +						{
   1.292 +						orientationOK = ETrue;
   1.293 +						}
   1.294 +					}
   1.295 +				if(orientationOK)
   1.296 +				    {
   1.297 +				    TBool tempElementFlip = aSurfaceConfiguration.Flip();
   1.298 +				    TBool tempCRPFlip = aElement.SourceFlipping();
   1.299 +				    if(tempElementFlip == tempCRPFlip)
   1.300 +				        {
   1.301 +				        match = ETrue;
   1.302 +				        }
   1.303 +				    }
   1.304 +				}
   1.305 +			}
   1.306 +		}
   1.307 +	return match;
   1.308 +	}
   1.309 +
   1.310 +/** Sets the surfaceconfiguration to the element.
   1.311 +
   1.312 +@param aElement The element to set.
   1.313 +@param aSurfaceConfiguration The configuration to set to the element.
   1.314 +@param aLimitedSet Set all values if EFalse, otherwise only set viewport and orientation.
   1.315 +@return KErrNone on success, error from compositor if it could not set the surface id.
   1.316 +*/
   1.317 +TInt CWindowElement::SetElement(MWsElement& aElement, const TSurfaceConfiguration& aSurfaceConfiguration, TBool aLimitedSet)
   1.318 +	{	//aLimitedSet - EFalse = set all . ETrue = set viewport and orientation only
   1.319 +	if (!aLimitedSet)
   1.320 +		{
   1.321 +		//set surface id
   1.322 +		TSurfaceId newSurfaceId;
   1.323 +		aSurfaceConfiguration.GetSurfaceId(newSurfaceId);
   1.324 +		TInt error = aElement.ConnectSurface(newSurfaceId);
   1.325 +		if (error != KErrNone)
   1.326 +			{
   1.327 +			return error;
   1.328 +			}
   1.329 +		
   1.330 +		//set extent
   1.331 +		TRect newExtent;
   1.332 +		aSurfaceConfiguration.GetExtent(newExtent);
   1.333 +		aElement.SetDestinationRectangle(newExtent);
   1.334 +		}
   1.335 +	
   1.336 +	//set viewport
   1.337 +	TRect newViewport;
   1.338 +	aSurfaceConfiguration.GetViewport(newViewport);
   1.339 +	if (newViewport.iTl.iX < 0 || newViewport.iTl.iY < 0)
   1.340 +		{
   1.341 +		return KErrArgument;
   1.342 +		}
   1.343 +	
   1.344 +	if (!newViewport.IsEmpty())
   1.345 +		{	//need to check if viewport is valid size
   1.346 +		TInt error = aElement.SetSourceRectangle(newViewport);
   1.347 +        if (error != KErrNone)
   1.348 +            {
   1.349 +            return error;
   1.350 +            }
   1.351 +		}
   1.352 +
   1.353 +	//set orientation
   1.354 +	//Note for compatibility with Oghma:
   1.355 +	//This method does not properly error check or return KErrArgument,
   1.356 +	//as that would propagate a client panic in SetBackgroundSurface which did not previously occur.
   1.357 +	aElement.SetSourceRotation(GcToElementRotation(aSurfaceConfiguration.Orientation()));
   1.358 +	// Set or clear flip if the element flags are changing
   1.359 +	if ( (!aSurfaceConfiguration.Flip()) != (!aElement.SourceFlipping()) ) 
   1.360 +	    {
   1.361 +        aElement.SetSourceFlipping(!aElement.SourceFlipping());
   1.362 +	    }
   1.363 +	return KErrNone;
   1.364 +	}
   1.365 +
   1.366 +/** Attempts to insert a region cache in to the the windows iCache.
   1.367 +If it fails, it will remove all cache beyond its index as it is now out of step.
   1.368 +
   1.369 +@param aIndexToInsert The index of where to insert the cache.
   1.370 +@return ETrue on success, else EFalse.
   1.371 +*/
   1.372 +
   1.373 +TBool CWindowElement::InsertCache(TInt aIndexToInsert)
   1.374 +	{
   1.375 +	RRegion newRegion;
   1.376 +	TInt cacheInsertError = iCache.Insert(newRegion, aIndexToInsert);
   1.377 +	if (cacheInsertError == KErrNone)
   1.378 +		{
   1.379 +		newRegion.Close(); 
   1.380 +		return ETrue;	//inserted cache
   1.381 +		}
   1.382 +	if (iCache.Count() > aIndexToInsert)
   1.383 +		{	//if insert failed and there is more cache above where its trying to insert
   1.384 +		iCache.Remove(iCache.Count()-1);
   1.385 +		cacheInsertError = iCache.Insert(newRegion, aIndexToInsert);
   1.386 +		
   1.387 +		if (cacheInsertError == KErrNone)
   1.388 +			{
   1.389 +			newRegion.Close();
   1.390 +			return ETrue;	//inserted cache
   1.391 +			}
   1.392 +		}
   1.393 +	
   1.394 +	//cant insert this cache, there may be invalid cache to destroy
   1.395 +	TInt count = iCache.Count();
   1.396 +	while(count > aIndexToInsert)
   1.397 +		{	//if there is cache where we wished to insert, it and everything above is now invalid
   1.398 +		iCache[count-1].Close();
   1.399 +		iCache.Remove(count-1);
   1.400 +		count--;
   1.401 +		};
   1.402 +	newRegion.Close();
   1.403 +	return EFalse;	//failed to insert
   1.404 +	}
   1.405 +
   1.406 +/** Will send a notification that a surface being unreferenced if it is not
   1.407 +found in any WindowElementSet
   1.408 +
   1.409 +@pre Make sure the element being removed has already had its surfaceid set to NULL.
   1.410 +@param aSurfaceId The surface being removed
   1.411 +*/
   1.412 +void CWindowElement::NotifyReleasingSurface(TSurfaceId aSurfaceId)
   1.413 +	{
   1.414 +	if (CWsTop::SearchDuplicateSurfaceId(aSurfaceId) == EFalse)
   1.415 +		{	//NOT found other instance of surface id, send notification now
   1.416 +		TWservCrEvent crEvent(TWservCrEvent::ESurfaceUnreferenced, sizeof(aSurfaceId), &aSurfaceId);
   1.417 +		TWindowServerEvent::NotifyDrawer(crEvent);
   1.418 +		}
   1.419 +	}
   1.420 +
   1.421 +/** Called to place a element within the window.
   1.422 +It will either create a new element, recycle a element, or extend a element .
   1.423 +
   1.424 +@pre UnassignPlacedElements has already been called.
   1.425 +@param aPlacedAttributes Returns the attributes of the surface placed.
   1.426 +@param aSurfaceConfiguration The surface configuration for the surface to place.
   1.427 +@param aUserDefinedRegion The user defined clipping of the window.
   1.428 +@param aScene Access to the scene.
   1.429 +@return Serious error, otherwise flags to describe what extra work needs doing.
   1.430 +
   1.431 +@see CWindowElementSet::AssignPlacedElement
   1.432 +*/
   1.433 +TInt CWindowElement::AssignPlacedElement(
   1.434 +		TPlacedAttributes*& aPlacedAttributes,
   1.435 +		const TSurfaceConfiguration& aSurfaceConfiguration,
   1.436 +		const TRegion& aUserDefinedRegion,
   1.437 +		MWsScene& aScene,
   1.438 +		TInt	aGcDrawingCount
   1.439 +		)
   1.440 +	{	
   1.441 +	if (iStartGcDrawingCount!=aGcDrawingCount)
   1.442 +		{
   1.443 +		FlagAssignedElementsDrawnOver(aGcDrawingCount);
   1.444 +		}
   1.445 +
   1.446 +	if (iNonRedrawVisibleRegion.CheckError())
   1.447 +		{
   1.448 +		//CANNOT DO ANYTHING, LEAVE
   1.449 +		return KErrNotReady;	//error
   1.450 +		}
   1.451 +
   1.452 +	TRect surfaceExtent;
   1.453 +	aSurfaceConfiguration.GetExtent(surfaceExtent);
   1.454 +	if (!aUserDefinedRegion.Intersects(surfaceExtent))
   1.455 +		{
   1.456 +		//clippedRegion.Close();
   1.457 +		return KErrCancel;	//no intersection, so nothing to do
   1.458 +		}
   1.459 +
   1.460 +	TInt matchFound = -1;
   1.461 +	
   1.462 +	//check 'current' (last placed element) to see if it is a match
   1.463 +	if (iHighestReusedIndex != -1)
   1.464 +		{
   1.465 +		if (IsElementAMatch(*(iPlacedElements[iHighestReusedIndex].iElement),aSurfaceConfiguration,iPlacedElements[iHighestReusedIndex].Flags()) )
   1.466 +			{
   1.467 +			matchFound = iHighestReusedIndex;
   1.468 +			}
   1.469 +		}
   1.470 +
   1.471 +	if (matchFound == -1)
   1.472 +		{
   1.473 +		//check current unassigned elements to see if there is a match
   1.474 +		for (TInt ii = (iHighestReusedIndex == -1)?0:iHighestReusedIndex+1; ii < iPlacedElements.Count(); ii++)
   1.475 +			{
   1.476 +			if (IsElementAMatch(*(iPlacedElements[ii].iElement),aSurfaceConfiguration,iPlacedElements[ii].Flags()))
   1.477 +				{
   1.478 +				matchFound = ii;
   1.479 +				break;
   1.480 +				}
   1.481 +			}
   1.482 +		}
   1.483 +	TInt returnValue = 0;
   1.484 +	if (matchFound == -1)
   1.485 +		{	//not found a element to extend
   1.486 +		//iResort = ETrue;
   1.487 +		matchFound = 0;
   1.488 +		while(1)
   1.489 +			{
   1.490 +			if (iPlacedElements.Count() == 0 || iHighestReusedIndex == -1 || matchFound == iHighestReusedIndex)	//reached last re assigned element, so cannot reuse one
   1.491 +				{
   1.492 +				matchFound = -1;
   1.493 +				break;
   1.494 +				}
   1.495 +			if (iPlacedElements[matchFound].Unassigned() && !iPlacedElements[matchFound].LockedConfig())	//found a element to reuse
   1.496 +				{
   1.497 +				break;
   1.498 +				}
   1.499 +			matchFound++;
   1.500 +			};
   1.501 +
   1.502 +		if (matchFound != -1)
   1.503 +			{	//found a element to recycle
   1.504 +			returnValue = AssignRecycleElement(matchFound,aSurfaceConfiguration,aUserDefinedRegion,aScene);
   1.505 +			}
   1.506 +		else
   1.507 +			{	//need to create a new element
   1.508 +			returnValue = AssignCreateElement(aSurfaceConfiguration,aUserDefinedRegion,aScene);
   1.509 +			}
   1.510 +		}
   1.511 +	else
   1.512 +		{	//found a candidate to extend
   1.513 +		returnValue = AssignExtendElement(matchFound,aSurfaceConfiguration,aUserDefinedRegion,aScene);
   1.514 +		}
   1.515 +	if (returnValue < KErrNone)
   1.516 +		{	//failed with error
   1.517 +			return returnValue;
   1.518 +		}
   1.519 +	aPlacedAttributes = &iPlacedElements[iHighestReusedIndex];
   1.520 +	iFlags|=returnValue;	//update window wide flags
   1.521 +	return returnValue;
   1.522 +	}
   1.523 +
   1.524 +/** Recycle a element assigned to the window.
   1.525 +
   1.526 +@param aIndex The index of the placed element to recycle.
   1.527 +@param aSurfaceConfiguration The surface configuration for the surface to place.
   1.528 +@param aUserDefinedRegion The user defined clipping of the window.
   1.529 +@param aScene Access to the scene.
   1.530 +@return Serious error, otherwise flags to describe what extra work needs doing.
   1.531 +
   1.532 +@see CWindowElement::AssignPlacedElement
   1.533 +*/
   1.534 +TInt CWindowElement::AssignRecycleElement(const TInt aIndex, const TSurfaceConfiguration& aSurfaceConfiguration, const TRegion& aUserDefinedRegion, MWsScene& aScene)
   1.535 +	{
   1.536 +	TInt returnValue = 0;
   1.537 +	TInt error = KErrNone;
   1.538 +	STACK_REGION clippedRegion;
   1.539 +	TRect surfaceExtent;
   1.540 +	aSurfaceConfiguration.GetExtent(surfaceExtent);
   1.541 +	clippedRegion.AddRect(surfaceExtent);
   1.542 +	clippedRegion.Intersect(aUserDefinedRegion);
   1.543 +	if (clippedRegion.CheckError())
   1.544 +		{	//failed crucial calculation, leave while nothing has been changed!
   1.545 +		clippedRegion.Close();
   1.546 +		return KErrNoMemory;
   1.547 +		}
   1.548 +	WS_ASSERT_DEBUG(!clippedRegion.IsEmpty(),EWsPanicRegion);
   1.549 +	if (clippedRegion.IsEmpty())
   1.550 +		{
   1.551 +		return KErrArgument;
   1.552 +		}
   1.553 +	
   1.554 +	//keep note of surface being removed
   1.555 +	TSurfaceId oldSurfaceId = iPlacedElements[aIndex].iElement->ConnectedSurface();
   1.556 +	
   1.557 +	//set element with new attributes
   1.558 +	error = SetElement(*(iPlacedElements[aIndex].iElement),aSurfaceConfiguration,EFalse);
   1.559 +	
   1.560 +	if (error < KErrNone)
   1.561 +		{	//must be a bad surface id or bad viewport, leave while nothing has been changed!
   1.562 +		clippedRegion.Close();
   1.563 +		return error;
   1.564 +		}
   1.565 +	
   1.566 +	returnValue|=EFastPath;
   1.567 +	returnValue|=EPauseComposition;
   1.568 +	//element has been altered, must make sure old surface unreferenced notify will be called
   1.569 +	if (!iRemovedSurfacesValid)
   1.570 +		{	//didnt reserve space for surface ids, do search immediately
   1.571 +		NotifyReleasingSurface(oldSurfaceId);
   1.572 +		}
   1.573 +	else
   1.574 +		{	//keep memory of surface id - search will be done during cleanup. Space already been reserved
   1.575 +		iRemovedSurfaces.Append(oldSurfaceId);
   1.576 +		}
   1.577 +
   1.578 +	//order placed element correctly in iPlacedElements
   1.579 +	TPlacedAttributes tempAttributes = iPlacedElements[aIndex];
   1.580 +	iPlacedElements.Remove(aIndex);
   1.581 +	//keep cache up to date
   1.582 +	if (iCache.Count() > aIndex)
   1.583 +		{
   1.584 +		iCache[aIndex].Close();
   1.585 +		iCache.Remove(aIndex);
   1.586 +		}
   1.587 +
   1.588 +	WS_ASSERT_DEBUG(iHighestReusedIndex != -1,EWsPanicRecycleElement);
   1.589 +	if (iHighestReusedIndex == -1)
   1.590 +		{
   1.591 +		return KErrArgument;
   1.592 +		}
   1.593 +	
   1.594 +	error = iPlacedElements.Insert(tempAttributes, iHighestReusedIndex);
   1.595 +	if (error < KErrNone)
   1.596 +		{	//must destroy the element and leave.  Old elements surface unregistration already dealt with
   1.597 +		aScene.DestroySceneElement(tempAttributes.iElement);
   1.598 +		clippedRegion.Close();
   1.599 +		return KErrNoMemory;
   1.600 +		}
   1.601 +	returnValue|=EResort;	//element has been moved, will need a resort
   1.602 +	//set placed element attributes
   1.603 +	iPlacedElements[iHighestReusedIndex].SetUnassigned(EFalse);
   1.604 +	iPlacedElements[iHighestReusedIndex].SetLockedConfig(ETrue);
   1.605 +	iPlacedElements[iHighestReusedIndex].SetChangedClip(ETrue);
   1.606 +	
   1.607 +	TRect tempViewport;
   1.608 +	aSurfaceConfiguration.GetViewport(tempViewport);
   1.609 +
   1.610 +	TBool cacheSaved = EFalse;
   1.611 +	if (iCache.Count() >= iHighestReusedIndex)
   1.612 +		{	//need to update cache with visible region of element
   1.613 +		TBool insertSuccess = InsertCache(iHighestReusedIndex);
   1.614 +		if (insertSuccess != EFalse)
   1.615 +			{
   1.616 +			iCache[iHighestReusedIndex].Copy(clippedRegion);
   1.617 +			if (!iCache[iHighestReusedIndex].CheckError())
   1.618 +				{
   1.619 +				cacheSaved = ETrue;
   1.620 +				}
   1.621 +			}
   1.622 +		}
   1.623 +	if (cacheSaved == EFalse)
   1.624 +		{	//need to set clipping to element immediately
   1.625 +		TInt count = iCache.Count();
   1.626 +		while(count > iHighestReusedIndex)
   1.627 +			{	//if there is cache where we wished to insert, it and everything above is now invalid
   1.628 +			iCache[count-1].Close();
   1.629 +			iCache.Remove(count-1);
   1.630 +			count--;
   1.631 +			};
   1.632 +		}
   1.633 +	clippedRegion.Close();
   1.634 +	return returnValue;
   1.635 +	}
   1.636 +
   1.637 +/** Create a new element for the placed surface.
   1.638 +
   1.639 +@param aSurfaceConfiguration The surface configuration for the surface to place.
   1.640 +@param aUserDefinedRegion The user defined clipping of the window.
   1.641 +@param aScene Access to the scene.
   1.642 +@return Serious error, otherwise flags to describe what extra work needs doing.
   1.643 +
   1.644 +@see CWindowElement::AssignPlacedElement
   1.645 +*/
   1.646 +TInt CWindowElement::AssignCreateElement(const TSurfaceConfiguration& aSurfaceConfiguration, const TRegion& aUserDefinedRegion, MWsScene& aScene)
   1.647 +	{
   1.648 +	TInt returnValue = 0;
   1.649 +	TInt error = KErrNone;
   1.650 +	STACK_REGION clippedRegion;
   1.651 +	TRect surfaceExtent;
   1.652 +	aSurfaceConfiguration.GetExtent(surfaceExtent);
   1.653 +	//calculate region
   1.654 +	clippedRegion.AddRect(surfaceExtent);
   1.655 +	clippedRegion.Intersect(aUserDefinedRegion);
   1.656 +	if (clippedRegion.CheckError())
   1.657 +		{	//failed, return KErrNoMemory
   1.658 +		clippedRegion.Close();
   1.659 +		return KErrNoMemory;
   1.660 +		}
   1.661 +	
   1.662 +	TPlacedAttributes newElement;
   1.663 +	if (iHighestReusedIndex == -1)
   1.664 +		{
   1.665 +		error = iPlacedElements.Insert(newElement,0);
   1.666 +		}
   1.667 +	else
   1.668 +		{	//insert above current highest reused element
   1.669 +		error = iPlacedElements.Insert(newElement,iHighestReusedIndex+1);
   1.670 +		}
   1.671 +	if (error >= KErrNone)
   1.672 +		{
   1.673 +		iHighestReusedIndex++;
   1.674 +		}
   1.675 +	else
   1.676 +		{
   1.677 +		//FAILED! LEAVE FUNCTION. nothing has been changed
   1.678 +		clippedRegion.Close();
   1.679 +		return KErrNoMemory;
   1.680 +		}
   1.681 +
   1.682 +	//create new element with new configuration
   1.683 +	MWsElement* element = NULL;
   1.684 +	TRAP(error,element = aScene.CreateSceneElementL());
   1.685 +	if (error != KErrNone)
   1.686 +		{
   1.687 +		//FAILED! LEAVE FUNCTION. nothing has been changed
   1.688 +		clippedRegion.Close();
   1.689 +		return error;
   1.690 +		}
   1.691 +
   1.692 +	element->SetGlobalAlpha( 0xFF );
   1.693 +
   1.694 +	iPlacedElements[iHighestReusedIndex].iElement = element;
   1.695 +	error = SetElement(*element,aSurfaceConfiguration,EFalse);
   1.696 +	if (error < KErrNone)
   1.697 +		{
   1.698 +		//must be a bad surface id or viewport. Leave now.
   1.699 +		aScene.DestroySceneElement(element);
   1.700 +		clippedRegion.Close();
   1.701 +		iPlacedElements.Remove(iHighestReusedIndex);
   1.702 +		iHighestReusedIndex--;
   1.703 +		return error;
   1.704 +		}
   1.705 +	
   1.706 +	//Now certain that these must be flagged
   1.707 +	returnValue|=EFastPath;
   1.708 +	returnValue|=EPauseComposition;
   1.709 +	returnValue|=EResort;
   1.710 +
   1.711 +	//set placed element attributes
   1.712 +	iPlacedElements[iHighestReusedIndex].SetUnassigned(EFalse);
   1.713 +	iPlacedElements[iHighestReusedIndex].SetChangedClip(ETrue);
   1.714 +	iPlacedElements[iHighestReusedIndex].SetLockedConfig(ETrue);
   1.715 +	
   1.716 +	TRect tempViewport;
   1.717 +	aSurfaceConfiguration.GetViewport(tempViewport);
   1.718 +
   1.719 +	TBool cacheSaved = EFalse;
   1.720 +	if (iCache.Count() >= iHighestReusedIndex)
   1.721 +		{	//should try to add new elements region to cache
   1.722 +		TBool insertSuccess = InsertCache(iHighestReusedIndex);
   1.723 +		if (insertSuccess != EFalse)
   1.724 +			{
   1.725 +			iCache[iHighestReusedIndex].Copy(clippedRegion);
   1.726 +			if (!iCache[iHighestReusedIndex].CheckError())
   1.727 +				{
   1.728 +				cacheSaved = ETrue;
   1.729 +				}
   1.730 +			}
   1.731 +		}
   1.732 +	if (cacheSaved == EFalse)
   1.733 +		{	//need to clear cache from this point onward, and set clipping to element immediately
   1.734 +		TInt count = iCache.Count();
   1.735 +		while(count > iHighestReusedIndex)
   1.736 +			{	//if there is cache where we wished to insert, it and everything above is now invalid
   1.737 +			iCache[count-1].Close();
   1.738 +			iCache.Remove(count-1);
   1.739 +			count--;
   1.740 +			};
   1.741 +		}
   1.742 +	//add the new element to the scene
   1.743 +    error = aScene.InsertSceneElement(element, NULL);    //place at the back. will get ordered correctly later
   1.744 +	WS_ASSERT_DEBUG(error == KErrNone,EWsPanicSceneErrorIgnored);
   1.745 +	clippedRegion.Close();
   1.746 +	return returnValue;
   1.747 +	}
   1.748 +
   1.749 +/** Extend an existing placed surface element.
   1.750 +
   1.751 +@param The index of the element to extend.
   1.752 +@param aSurfaceConfiguration The surface configuration for the surface to place.
   1.753 +@param aUserDefinedRegion The user defined clipping of the window.
   1.754 +@param aScene Access to the scene.
   1.755 +@return Serious error, otherwise flags to describe what extra work needs doing.
   1.756 +
   1.757 +@see CWindowElement::AssignPlacedElement
   1.758 +*/
   1.759 +TInt CWindowElement::AssignExtendElement(const TInt aIndex, const TSurfaceConfiguration& aSurfaceConfiguration, const TRegion& aUserDefinedRegion, MWsScene& /*aScene*/)
   1.760 +	{
   1.761 +	TInt returnValue = 0;
   1.762 +	TInt error = KErrNone;
   1.763 +	STACK_REGION backupRegionAgainstFail;
   1.764 +	STACK_REGION clippedRegion;
   1.765 +	TRect surfaceExtent;
   1.766 +	aSurfaceConfiguration.GetExtent(surfaceExtent);
   1.767 +	TRect surfaceViewport;
   1.768 +	aSurfaceConfiguration.GetViewport(surfaceViewport);
   1.769 +	
   1.770 +	if (!iPlacedElements[aIndex].LockedConfig())
   1.771 +		{	//set element now. If viewport is invalid its ok to fail now
   1.772 +		error = SetElement(*iPlacedElements[aIndex].iElement,aSurfaceConfiguration,ETrue);
   1.773 +		if (error < KErrNone)
   1.774 +			{
   1.775 +			backupRegionAgainstFail.Close();
   1.776 +			clippedRegion.Close();
   1.777 +			return error;
   1.778 +			}
   1.779 +		}
   1.780 +
   1.781 +	if(iPlacedElements[aIndex].Unassigned() && !iPlacedElements[aIndex].Overlapping())
   1.782 +		{
   1.783 +		//backup stays clear
   1.784 +		clippedRegion.Copy(aUserDefinedRegion);
   1.785 +		}
   1.786 +	else
   1.787 +		{
   1.788 +		clippedRegion.Copy(iNonRedrawVisibleRegion);
   1.789 +		if (iCache.Count() > aIndex)	//if cache is ok
   1.790 +			{
   1.791 +			clippedRegion.Intersect(iCache[aIndex]);
   1.792 +			clippedRegion.Union(iCache[aIndex]);	//previously added regions should be kept
   1.793 +			}
   1.794 +		else
   1.795 +			{
   1.796 +			STACK_REGION tempSurfaceBaseArea;
   1.797 +            TRect destinationRect;
   1.798 +            iPlacedElements[aIndex].iElement->GetDestinationRectangle(destinationRect);
   1.799 +            tempSurfaceBaseArea.AddRect(destinationRect);
   1.800 +			tempSurfaceBaseArea.Offset(surfaceExtent.iTl);
   1.801 +			clippedRegion.Intersect(tempSurfaceBaseArea);
   1.802 +			clippedRegion.Union(tempSurfaceBaseArea);
   1.803 +			tempSurfaceBaseArea.Close();
   1.804 +			}
   1.805 +		backupRegionAgainstFail.Copy(clippedRegion);
   1.806 +		//clipped region should contain something
   1.807 +		clippedRegion.Union(aUserDefinedRegion);
   1.808 +		}
   1.809 +	clippedRegion.ClipRect(surfaceExtent);
   1.810 +	
   1.811 +	if (backupRegionAgainstFail.CheckError() || clippedRegion.CheckError())
   1.812 +		{	//failed critical calculations, leave now before anything has been changed
   1.813 +		clippedRegion.Close();
   1.814 +		backupRegionAgainstFail.Close();
   1.815 +		return KErrNoMemory;
   1.816 +		}
   1.817 +
   1.818 +	TBool setCache = EFalse;
   1.819 +	if (iCache.Count() > aIndex)	//if Cache is ok
   1.820 +		{
   1.821 +		//compare
   1.822 +		TInt compareReturn;
   1.823 +		compareReturn = TRegionExtend::Cast(iCache[aIndex]).TestDifference(clippedRegion);
   1.824 +
   1.825 +		if (compareReturn&TRegionExtend::EDiffers)	//element has changed
   1.826 +			{
   1.827 +			iPlacedElements[aIndex].SetChangedClip(ETrue);
   1.828 +			if (compareReturn&TRegionExtend::EAdd)	//element has become revealed on the screen
   1.829 +				{
   1.830 +				returnValue|=EPauseComposition;	//need to pause before the end of assign placed element
   1.831 +				}
   1.832 +			else
   1.833 +				{
   1.834 +				returnValue|=EFastPath;	//fastpath and pause will be called during cleanup
   1.835 +				}
   1.836 +			}
   1.837 +		
   1.838 +		//copy clipped region to cache
   1.839 +		iCache[aIndex].Copy(clippedRegion);	//can fail
   1.840 +		setCache = ETrue;
   1.841 +		if (iCache[aIndex].CheckError())
   1.842 +			{	//copy failed, remove cache from this element onwards
   1.843 +			TInt count = iCache.Count();
   1.844 +			while(count > aIndex)
   1.845 +				{	//if there is cache where we wished to insert, it and everything above is now invalid
   1.846 +				iCache[count-1].Close();
   1.847 +				iCache.Remove(count-1);
   1.848 +				count--;
   1.849 +				};
   1.850 +			setCache = EFalse;
   1.851 +			}
   1.852 +		}
   1.853 +	if (setCache == EFalse)
   1.854 +		{	//need to pause composition and update element immediately
   1.855 +		returnValue|=EPauseComposition;
   1.856 +		}
   1.857 +	iPlacedElements[aIndex].SetUnassigned(EFalse);
   1.858 +	iPlacedElements[aIndex].SetLockedConfig(ETrue);
   1.859 +	iHighestReusedIndex = aIndex;
   1.860 +	backupRegionAgainstFail.Close();
   1.861 +	clippedRegion.Close();
   1.862 +	return returnValue;
   1.863 +	}
   1.864 +
   1.865 +/** Called at the end of a redraw to set cached changed to elements, and
   1.866 +removed elements that are still unassigned.
   1.867 +
   1.868 +@param aScene Access to the scene.
   1.869 +@return Flags to describe what extra work needs doing.
   1.870 +@see CWindowElementSet::CleanUpPlacedElements
   1.871 +*/
   1.872 +TInt CWindowElement::CleanUpPlacedElements(MWsScene& aScene, TInt aGcDrawingCount)
   1.873 +	{
   1.874 +	if (iStartGcDrawingCount!=aGcDrawingCount)
   1.875 +		{
   1.876 +		FlagAssignedElementsDrawnOver(aGcDrawingCount);
   1.877 +		}
   1.878 +	TInt returnFlags = iFlags;
   1.879 +	if (iFlags > 0)
   1.880 +		{	//copy flags to return, and ensure pause is flagged
   1.881 +		returnFlags|=EPauseComposition;
   1.882 +		}
   1.883 +	TInt ii;
   1.884 +	TBool removeElements = EFalse;
   1.885 +	//destroy placed element reference
   1.886 +	for (ii=0; ii<iPlacedElements.Count(); ii++)
   1.887 +		{
   1.888 +		if (iPlacedElements[ii].Unassigned() && !iPlacedElements[ii].Overlapping())
   1.889 +			{
   1.890 +			returnFlags|=EFastPath;	//need to invoke 1108
   1.891 +			if (iRemovedSurfacesValid)
   1.892 +				{
   1.893 +				iRemovedSurfaces.Append(iPlacedElements[ii].iElement->ConnectedSurface());
   1.894 +				iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
   1.895 +				}
   1.896 +			else
   1.897 +				{
   1.898 +				TSurfaceId tempId = iPlacedElements[ii].iElement->ConnectedSurface();
   1.899 +				iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
   1.900 +				NotifyReleasingSurface(tempId);
   1.901 +				}
   1.902 +			removeElements = ETrue;
   1.903 +			}
   1.904 +		}
   1.905 +
   1.906 +	if (removeElements)
   1.907 +		{	//there will be elements to remove
   1.908 +		for (ii=0; ii<iPlacedElements.Count(); ii++)
   1.909 +			{
   1.910 +			if (iPlacedElements[ii].iElement->ConnectedSurface().IsNull())
   1.911 +				{
   1.912 +				aScene.DestroySceneElement(iPlacedElements[ii].iElement);
   1.913 +				iPlacedElements.Remove(ii);
   1.914 +				if (iCache.Count() > ii)
   1.915 +					{	//there is cache for this element, remove it
   1.916 +					iCache[ii].Close();
   1.917 +					iCache.Remove(ii);
   1.918 +					}
   1.919 +				ii--;
   1.920 +				}
   1.921 +			}
   1.922 +		}
   1.923 +
   1.924 +	if (iRemovedSurfaces.Count() > 1)
   1.925 +		{	//remove duplicates of surface id from removal list
   1.926 +		for (ii = 0; ii < iRemovedSurfaces.Count() - 1; ii++)
   1.927 +			{
   1.928 +			for (TInt jj = ii+1; jj < iRemovedSurfaces.Count(); jj++)
   1.929 +				{
   1.930 +				if (iRemovedSurfaces[ii] == iRemovedSurfaces[jj])
   1.931 +					{
   1.932 +					iRemovedSurfaces.Remove(jj);
   1.933 +					jj--;
   1.934 +					}
   1.935 +				}
   1.936 +			}
   1.937 +		}
   1.938 +	
   1.939 +	//do global search for surface ids
   1.940 +	while (iRemovedSurfaces.Count()>0)
   1.941 +		{
   1.942 +		TInt tempRemoval = iRemovedSurfaces.Count() - 1;
   1.943 +		NotifyReleasingSurface(iRemovedSurfaces[tempRemoval]);
   1.944 +		iRemovedSurfaces.Remove(tempRemoval);
   1.945 +		};
   1.946 +
   1.947 +	iRemovedSurfaces.Reset();
   1.948 +
   1.949 +	//clip unassigned overlapping entries, mark as changedclip if region changed
   1.950 +	for (ii=0; ii<iPlacedElements.Count(); ii++)
   1.951 +		{
   1.952 +		if (iPlacedElements[ii].Unassigned())	//non overlapping unassigned have already been destroyed
   1.953 +			{	//these must overlap
   1.954 +			TBool failureOccured = EFalse;
   1.955 +			if (iCache.Count() > ii)
   1.956 +				{
   1.957 +				//if cache region is entirely inside non redraw vis region, nothing to update!
   1.958 +				TInt changed = 
   1.959 +						TRegionExtend::Cast(iCache[ii]).TestDifference(iNonRedrawVisibleRegion);
   1.960 +				if (changed&TRegionExtend::ESub)
   1.961 +					{	//the clipping will change
   1.962 +					iCache[ii].Intersect(iNonRedrawVisibleRegion);
   1.963 +					if (iCache[ii].CheckError())
   1.964 +						{
   1.965 +						failureOccured = ETrue;
   1.966 +						iCache[ii].Close();
   1.967 +						iCache.Remove(ii);
   1.968 +						}
   1.969 +					else
   1.970 +						{
   1.971 +						iPlacedElements[ii].SetChangedClip(ETrue);	//changed clipping
   1.972 +						}
   1.973 +					}
   1.974 +				}
   1.975 +			else
   1.976 +				{	//attempt getting element region to perform calculation and update element directly
   1.977 +				STACK_REGION elementRegion;
   1.978 +	            TRect destinationRect;
   1.979 +	            iPlacedElements[ii].iElement->GetDestinationRectangle(destinationRect);
   1.980 +	            elementRegion.AddRect(destinationRect);
   1.981 +				if (elementRegion.CheckError())
   1.982 +					{
   1.983 +					failureOccured = ETrue;
   1.984 +					}
   1.985 +				else
   1.986 +					{
   1.987 +					TRect elementExtent;
   1.988 +					iPlacedElements[ii].iElement->GetDestinationRectangle(elementExtent);
   1.989 +					elementRegion.Offset(elementExtent.iTl);	//get element region into absolute coordinates
   1.990 +					TInt changed =
   1.991 +						TRegionExtend::Cast(elementRegion).TestDifference(iNonRedrawVisibleRegion);
   1.992 +					if (changed&TRegionExtend::ESub)
   1.993 +						{	//need to clip element back
   1.994 +						elementRegion.Intersect(iNonRedrawVisibleRegion);
   1.995 +						if (!elementRegion.CheckError())
   1.996 +							{
   1.997 +							elementRegion.Offset(-elementExtent.iTl);	//put element region back into relative coords
   1.998 +							returnFlags|=EFastPath;
   1.999 +							}
  1.1000 +						else
  1.1001 +							{
  1.1002 +							failureOccured = ETrue;
  1.1003 +							}
  1.1004 +						}
  1.1005 +					}
  1.1006 +				elementRegion.Close();
  1.1007 +				}
  1.1008 +			if (failureOccured)
  1.1009 +				{	//need to release element
  1.1010 +				TSurfaceId tempId = iPlacedElements[ii].iElement->ConnectedSurface();
  1.1011 +				iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
  1.1012 +				NotifyReleasingSurface(tempId);
  1.1013 +				aScene.DestroySceneElement(iPlacedElements[ii].iElement);
  1.1014 +				returnFlags|=EFailed;
  1.1015 +				iPlacedElements.Remove(ii);	//remove placed element entry, cache is already removed
  1.1016 +				ii--;
  1.1017 +				}
  1.1018 +			}
  1.1019 +		}
  1.1020 +
  1.1021 +	//update any elements marked as changedclip , clear all cache, clear flags
  1.1022 +	for (ii=0; ii<iPlacedElements.Count(); ii++)
  1.1023 +		{
  1.1024 +		if (iPlacedElements[ii].ChangedClip())
  1.1025 +			{
  1.1026 +			returnFlags|=EFastPath;	//need to invoke 1108
  1.1027 +			}
  1.1028 +		else
  1.1029 +			if (	!(returnFlags&EFastPath)
  1.1030 +				&&	iPlacedElements[ii].LockedConfig() && !iPlacedElements[ii].Unassigned()
  1.1031 +				&&	iPlacedElements[ii].DrawnOver()!=iPlacedElements[ii].DrawnOverLast()	)
  1.1032 +						{
  1.1033 +						returnFlags|=EFastPath;
  1.1034 +						}
  1.1035 +		iPlacedElements[ii].ClearRedrawFlags();
  1.1036 +		if (iCache.Count() > ii)
  1.1037 +			{
  1.1038 +			iCache[ii].Clear();
  1.1039 +			}
  1.1040 +		}
  1.1041 +	iCache.Reset();
  1.1042 +	iNonRedrawVisibleRegion.Clear();
  1.1043 +	iHighestReusedIndex = -1;
  1.1044 +	return returnFlags;
  1.1045 +	}
  1.1046 +
  1.1047 +/** Move all elements in this windowelement to above the supplied element in the scene
  1.1048 +
  1.1049 +@param aScene Access to the scene.
  1.1050 +@param aElement The element to set all this windows elements above.
  1.1051 +*/
  1.1052 +
  1.1053 +void CWindowElement::MoveToAboveGivenElement(MWsScene& aScene, MWsElement* aElement)
  1.1054 +	{
  1.1055 +	MWsElement* newAboveElement = iBackgroundElement.iElement;
  1.1056 +	MWsElement* previousAboveElement = aElement;
  1.1057 +	if (newAboveElement)
  1.1058 +		{
  1.1059 +		aScene.InsertSceneElement(newAboveElement, previousAboveElement);	//background element is put on first
  1.1060 +		previousAboveElement = newAboveElement;		
  1.1061 +		}
  1.1062 +	TInt placedElements = iPlacedElements.Count();
  1.1063 +	if (placedElements != 0)
  1.1064 +		{
  1.1065 +		for (TInt ii = 0; ii < placedElements; ii++)
  1.1066 +			{
  1.1067 +			newAboveElement = iPlacedElements[ii].iElement;
  1.1068 +			aScene.InsertSceneElement(newAboveElement, previousAboveElement);	//place element above previous above element
  1.1069 +			previousAboveElement = newAboveElement;	
  1.1070 +			}
  1.1071 +		}
  1.1072 +	}
  1.1073 +
  1.1074 +/** Updates the elements extent, whether from a window movement or a change in window size
  1.1075 +
  1.1076 +@param aOffset The movement of the window.  If NULL then the window has changed size.
  1.1077 +@see CWindowElementSet::UpdateElementExtent
  1.1078 +*/
  1.1079 +void CWindowElement::UpdateElementExtent(const TPoint* aOffset)
  1.1080 +	{
  1.1081 +	if (aOffset)	//window moved
  1.1082 +		{
  1.1083 +		TRect tempExtent;
  1.1084 +		MWsElement* element = iBackgroundElement.iElement;
  1.1085 +		TBool complete = EFalse;
  1.1086 +		TInt placedElementDone = -1;
  1.1087 +		while (!complete)
  1.1088 +			{
  1.1089 +			if (!element && placedElementDone < iPlacedElements.Count()-1)
  1.1090 +				{
  1.1091 +				placedElementDone++;
  1.1092 +				element = iPlacedElements[placedElementDone].iElement;
  1.1093 +				}
  1.1094 +			if (!element)
  1.1095 +				{
  1.1096 +				complete = ETrue;
  1.1097 +				}
  1.1098 +			else
  1.1099 +				{
  1.1100 +				tempExtent = const_cast<CWsClientWindow&>(iWindow).GetOriginalDestElementRect();
  1.1101 +				tempExtent.Move(*aOffset);
  1.1102 +				element->SetDestinationRectangle(tempExtent);
  1.1103 +				const_cast<CWsClientWindow&>(iWindow).SetOriginalDestElementRect(tempExtent);
  1.1104 +				}
  1.1105 +			element = NULL;
  1.1106 +			}
  1.1107 +		}
  1.1108 +	else			//window changed size
  1.1109 +		{
  1.1110 +		if (!iBackgroundElement.ExplicitExtent() && iBackgroundElement.iElement)
  1.1111 +			{
  1.1112 +			iBackgroundElement.iElement->SetDestinationRectangle(iWindow.FullRect());
  1.1113 +			const_cast<CWsClientWindow&>(iWindow).SetOriginalDestElementRect(iWindow.FullRect());
  1.1114 +			}
  1.1115 +		}
  1.1116 +	}
  1.1117 +
  1.1118 +/** Checks the windows placed elements when the windows visibility is changed.
  1.1119 +If the placed elements are no longer visible, they are removed.
  1.1120 +
  1.1121 +@param aRegion The new visible region of the window.
  1.1122 +@param aScene Access to the scene.
  1.1123 +@return ETrue if any elements have been removed, otherwise EFalse.
  1.1124 +@see CWindowElementSet::SetVisibleRegion
  1.1125 +*/
  1.1126 +TBool CWindowElement::SetVisibleRegion(const TRegion& aRegion, MWsScene& aScene)
  1.1127 +	{
  1.1128 +	TBool retcode=EFalse;
  1.1129 +	if (iPlacedElements.Count() == 0)
  1.1130 +		{
  1.1131 +		return EFalse;	//there is noting to do
  1.1132 +		}
  1.1133 +	
  1.1134 +	STACK_REGION tempRegion;
  1.1135 +	for (TInt ii = 0; ii < iPlacedElements.Count(); ii++)
  1.1136 +		{
  1.1137 +		TRect tempExtent;
  1.1138 +        tempRegion.Clear();
  1.1139 +        iPlacedElements[ii].iElement->GetDestinationRectangle(tempExtent);
  1.1140 +        tempRegion.AddRect(tempExtent);
  1.1141 +		if (tempRegion.CheckError())	//if there was error getting region
  1.1142 +			{
  1.1143 +			tempRegion.Clear();
  1.1144 +			tempRegion.AddRect(tempExtent);
  1.1145 +			}
  1.1146 +		else
  1.1147 +			{	//offset basearea of element
  1.1148 +			tempRegion.Offset(tempExtent.iTl);
  1.1149 +			}
  1.1150 +		TInt regionReturn = TRegionExtend::Cast(aRegion).TestDifference(tempRegion);
  1.1151 +		if (regionReturn&TRegionExtend::ENoIntersect)
  1.1152 +			{	//placed surface needs to be removed
  1.1153 +			TSurfaceId tempId = iPlacedElements[ii].iElement->ConnectedSurface();
  1.1154 +			iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
  1.1155 +			NotifyReleasingSurface(tempId);
  1.1156 +			aScene.DestroySceneElement(iPlacedElements[ii].iElement);
  1.1157 +			retcode = ETrue;	//a element has been removed
  1.1158 +			iPlacedElements.Remove(ii);
  1.1159 +			if (iCache.Count() > ii)
  1.1160 +				{	//keep cache up to date
  1.1161 +				iCache[ii].Close();
  1.1162 +				iCache.Remove(ii);
  1.1163 +				}
  1.1164 +			ii--;
  1.1165 +			}
  1.1166 +		}
  1.1167 +	tempRegion.Close();
  1.1168 +	return retcode;
  1.1169 +	}
  1.1170 +
  1.1171 +/** Checks if any of the windows element ids match the one in question.
  1.1172 +
  1.1173 +@param aSurfaceId The surface id to match against.
  1.1174 +@return ETrue if a match is found, otherwise EFalse.
  1.1175 +
  1.1176 +@see CWindowElementSet::SearchDuplicateSurfaceId
  1.1177 +*/
  1.1178 +TBool CWindowElement::SearchDuplicateSurfaceId(const TSurfaceId& aSurfaceId)
  1.1179 +	{
  1.1180 +	if (iBackgroundElement.iElement)
  1.1181 +		{
  1.1182 +		if (iBackgroundElement.iElement->ConnectedSurface() == aSurfaceId)
  1.1183 +			{
  1.1184 +			return ETrue;
  1.1185 +			}
  1.1186 +		}
  1.1187 +	if (iPlacedElements.Count() > 0)
  1.1188 +		{
  1.1189 +		for (TInt ii = 0; ii < iPlacedElements.Count(); ii++)
  1.1190 +			{
  1.1191 +			if (iPlacedElements[ii].iElement)	//if removed without reserved space for ids, could be a null element
  1.1192 +				{
  1.1193 +				if (iPlacedElements[ii].iElement->ConnectedSurface() == aSurfaceId)
  1.1194 +					{
  1.1195 +					return ETrue;
  1.1196 +					}
  1.1197 +				}
  1.1198 +			}
  1.1199 +		}
  1.1200 +	return EFalse;
  1.1201 +	}
  1.1202 +
  1.1203 +/** Sets the opacity for the background surface and if setting to 0 removes placed surfaces.
  1.1204 +
  1.1205 +@param aOpacity The opacity to set.
  1.1206 +@param aScene Access to the scene.
  1.1207 +@see CWindowElementSet::SetElementOpacity
  1.1208 +*/
  1.1209 +void CWindowElement::SetElementOpacity(TInt aOpacity, MWsScene& aScene)
  1.1210 +	{
  1.1211 +	if (iBackgroundElement.iElement)
  1.1212 +		{
  1.1213 +	    TUint32 flags = 0;
  1.1214 +	    iBackgroundElement.iElement->GetTargetRendererFlags(flags);
  1.1215 +	    flags |= MWsElement::EElementTransparencyGlobalAlpha;
  1.1216 +		iBackgroundElement.iElement->SetTargetRendererFlags(MWsElement::EElementTransparencyGlobalAlpha);
  1.1217 +		iBackgroundElement.iElement->SetGlobalAlpha(aOpacity);
  1.1218 +		iBackgroundElement.SetConcealed(aOpacity==0);
  1.1219 +		}
  1.1220 +	if (aOpacity == 0)
  1.1221 +		{
  1.1222 +		while (iPlacedElements.Count() > 0)
  1.1223 +			{	//remove any placed elements
  1.1224 +			TInt placedElement = iPlacedElements.Count()-1;
  1.1225 +			TSurfaceId tempId = iPlacedElements[placedElement].iElement->ConnectedSurface();
  1.1226 +			iPlacedElements[placedElement].iElement->ConnectSurface(TSurfaceId::CreateNullId());
  1.1227 +			NotifyReleasingSurface(tempId);
  1.1228 +			aScene.DestroySceneElement(iPlacedElements[placedElement].iElement);
  1.1229 +			iPlacedElements.Remove(placedElement);
  1.1230 +			};
  1.1231 +		}
  1.1232 +	}
  1.1233 +
  1.1234 +MWsElement* CWindowElement::Element() const
  1.1235 +    {
  1.1236 +    return iBackgroundElement.iElement;
  1.1237 +    }
  1.1238 +
  1.1239 +//
  1.1240 +// CWindowElementSet
  1.1241 +//
  1.1242 +
  1.1243 +/** Destroys the set of window-element pairs.
  1.1244 +*/
  1.1245 +CWindowElementSet::~CWindowElementSet()
  1.1246 +	{
  1.1247 +	ASSERT(iElements.Count() == 0);
  1.1248 +	iElements.Close();
  1.1249 +	}
  1.1250 +
  1.1251 +/** Basic NewL constructor
  1.1252 +@param aScene To allow access to the scene.
  1.1253 +@param aComposer To allow access to the composer.
  1.1254 +*/
  1.1255 +CWindowElementSet* CWindowElementSet::NewL(MWsScene& aScene)
  1.1256 +	{
  1.1257 +	CWindowElementSet* wls = new (ELeave) CWindowElementSet(aScene);
  1.1258 +	return wls;
  1.1259 +	}
  1.1260 +
  1.1261 +/** Sets a new background surface on the window supplied.
  1.1262 +Will remove any previous background surface.
  1.1263 +
  1.1264 +@param aWindow The window to place a background surface on.
  1.1265 +@return A reference to the new background surface attributes.
  1.1266 +*/
  1.1267 +TBackgroundAttributes& CWindowElementSet::AcquireBackgroundElementL(CWsClientWindow& aWindow)
  1.1268 +	{
  1.1269 +	// Find the matching element.
  1.1270 +	TInt index;
  1.1271 +	TInt result = FindEntry(aWindow, index);
  1.1272 +	MWsElement* element;
  1.1273 +
  1.1274 +	// If a background element is already associated with the window, then unregister
  1.1275 +	// the surface. Create and add a new element to scene.
  1.1276 +	// This will ensure that the element has default values.
  1.1277 +	if (result != KErrNotFound && iElements[index]->iBackgroundElement.iElement)
  1.1278 +	    {
  1.1279 +	    element = iElements[index]->iBackgroundElement.iElement;
  1.1280 +		TSurfaceId surface = element->ConnectedSurface();
  1.1281 +		element->ConnectSurface(TSurfaceId::CreateNullId());
  1.1282 +		UnregisterSurface(surface);
  1.1283 +		iScene.DestroySceneElement(element);
  1.1284 +	    }
  1.1285 +	if (index < 0)
  1.1286 +		{
  1.1287 +		index = 0;
  1.1288 +		}
  1.1289 +
  1.1290 +	// Allocate a new element and add it to the set and the scene
  1.1291 +	element = iScene.CreateSceneElementL();
  1.1292 +	
  1.1293 +	TInt returnCode;
  1.1294 +	if (result == KErrNotFound)
  1.1295 +		{
  1.1296 +		CWindowElement* winelement = new CWindowElement(aWindow);
  1.1297 +		if (!winelement)
  1.1298 +			{
  1.1299 +			iScene.DestroySceneElement(element);
  1.1300 +			User::Leave(KErrNoMemory);
  1.1301 +			}
  1.1302 +		returnCode = iElements.Insert(winelement, index);
  1.1303 +		if(returnCode != KErrNone)
  1.1304 +		    {
  1.1305 +		    delete winelement;
  1.1306 +		    iScene.DestroySceneElement(element);
  1.1307 +		    User::Leave(returnCode);
  1.1308 +		    }
  1.1309 +		}
  1.1310 +
  1.1311 +	// Use the element below the insertion point to decide where the element
  1.1312 +	// goes in the scene
  1.1313 +	returnCode = KErrNone;
  1.1314 +	if (index == 0)
  1.1315 +		{
  1.1316 +		returnCode = iScene.InsertSceneElement(element, NULL);
  1.1317 +		}
  1.1318 +	else
  1.1319 +		{
  1.1320 +		//Find highest element in window below
  1.1321 +		MWsElement* below;
  1.1322 +		TInt placedCount = iElements[index-1]->iPlacedElements.Count();
  1.1323 +		if (placedCount > 0)
  1.1324 +			{
  1.1325 +			below = iElements[index-1]->iPlacedElements[placedCount-1].iElement;
  1.1326 +			}
  1.1327 +		else
  1.1328 +			{	//below = above = background surface
  1.1329 +			below = iElements[index-1]->iBackgroundElement.iElement;
  1.1330 +			}
  1.1331 +		returnCode = iScene.InsertSceneElement(element, below);
  1.1332 +		}
  1.1333 +	
  1.1334 +	__ASSERT_DEBUG(returnCode == KErrNone, Panic(EWsPanicSceneErrorIgnored));
  1.1335 +
  1.1336 +	iElements[index]->iBackgroundElement.iElement = element;
  1.1337 +	return iElements[index]->iBackgroundElement;
  1.1338 +	}
  1.1339 +
  1.1340 +/** Removes the background element of the specified window.
  1.1341 +
  1.1342 +@param aWindow The window to remove the background element from.
  1.1343 +@param aWindowClosing ETrue if aWindow is in between closing state.
  1.1344 +@return KErrNone on success or a system-wide error code.
  1.1345 +*/
  1.1346 +TInt CWindowElementSet::ReleaseBackgroundElement(const CWsClientWindow& aWindow, TBool aWindowClosing /*= EFalse*/)
  1.1347 +	{
  1.1348 +	TInt index;
  1.1349 +
  1.1350 +	TInt err = FindEntry(aWindow, index, aWindowClosing);
  1.1351 +	__ASSERT_DEBUG(err != KErrNotFound, Panic(EWsPanicNoWindowElement));
  1.1352 +
  1.1353 +	if (err==KErrNone)
  1.1354 +		{
  1.1355 +		CWindowElement* winElement=iElements[index];
  1.1356 +		if (winElement->iPlacedElements.Count() == 0)
  1.1357 +			{	//destroy the window entry
  1.1358 +			DestroyWindowElementEntry(index);
  1.1359 +			}
  1.1360 +		else
  1.1361 +			{	// just destroy the background
  1.1362 +			if (winElement->iBackgroundElement.iElement)
  1.1363 +				{
  1.1364 +				winElement->iBackgroundElement.Destroy(iScene, ETrue);
  1.1365 +				}
  1.1366 +			}
  1.1367 +		}
  1.1368 +	return err;
  1.1369 +	}
  1.1370 +
  1.1371 +/** Destroys the element associated with this set of attributes
  1.1372 +
  1.1373 +@param aScene To allow access to the scene.
  1.1374 +@param aUnregister Whether to call unregister on the surface id.
  1.1375 +@return ETrue if a element existed to be destroyed, otherwise EFalse.
  1.1376 +*/
  1.1377 +TBool TAttributes::Destroy(MWsScene& aScene, TBool aUnregister)
  1.1378 +	{
  1.1379 +	MWsElement* element = iElement;
  1.1380 +	if (element)
  1.1381 +		{
  1.1382 +		TSurfaceId surface = element->ConnectedSurface();
  1.1383 +		element->ConnectSurface(TSurfaceId::CreateNullId());
  1.1384 +        if (aUnregister)
  1.1385 +            {
  1.1386 +            aScene.UnregisterSurface(surface);
  1.1387 +            }
  1.1388 +		aScene.DestroySceneElement(element);
  1.1389 +		iElement = NULL;
  1.1390 +		}
  1.1391 +	return element!=NULL;
  1.1392 +	}
  1.1393 +
  1.1394 +/** Destroys all elements associated with the window and removes the window from the element set
  1.1395 +
  1.1396 +@param aWinElementIndex The index of the window in the element set
  1.1397 +*/
  1.1398 +void CWindowElementSet::DestroyWindowElementEntry(const TInt aWinElementIndex)
  1.1399 +	{
  1.1400 +	CWindowElement* winElement=iElements[aWinElementIndex];
  1.1401 +	const CWsClientWindow& window = winElement->iWindow;
  1.1402 +	winElement->iBackgroundElement.Destroy(iScene, ETrue);
  1.1403 +	for (TInt placedIndex=0,maxindex=winElement->iPlacedElements.Count();placedIndex<maxindex;placedIndex++)
  1.1404 +		{
  1.1405 +		winElement->iPlacedElements[placedIndex].Destroy(iScene, EFalse);
  1.1406 +		}
  1.1407 +	winElement->iPlacedElements.Close();
  1.1408 +	delete winElement;
  1.1409 +	iElements.Remove(aWinElementIndex);
  1.1410 +	window.Redraw()->SetHasElement(EFalse);
  1.1411 +	window.Screen()->ElementRemoved();
  1.1412 +	}
  1.1413 +
  1.1414 +/** Removes all elements in the associated window.
  1.1415 +
  1.1416 +@param aWindow The window to remove the elements for.
  1.1417 +@param aWindowClosing ETrue if aWindow is in between closing state.
  1.1418 +@return KErrNone on success, KErrNotFound if there was no windowelement entry.
  1.1419 +*/
  1.1420 +TInt CWindowElementSet::ReleaseAllElements(const CWsClientWindow& aWindow, TBool aWindowClosing /*= EFalse*/)
  1.1421 +	{
  1.1422 +	TInt index;
  1.1423 +
  1.1424 +	TInt err = FindEntry(aWindow, index, aWindowClosing);
  1.1425 +	__ASSERT_DEBUG(err != KErrNotFound, Panic(EWsPanicNoWindowElement));
  1.1426 +	if (err==KErrNone)
  1.1427 +		{
  1.1428 +		DestroyWindowElementEntry(index);
  1.1429 +		}
  1.1430 +	return err;
  1.1431 +	}
  1.1432 +
  1.1433 +/** For every window element, checks if any of the windows element ids match the one in question.
  1.1434 +
  1.1435 +@param aSurfaceId The surface id to match against.
  1.1436 +@return ETrue if a match is found, otherwise EFalse.
  1.1437 +
  1.1438 +@see CWindowElement::SearchDuplicateSurfaceId
  1.1439 +*/
  1.1440 +TBool CWindowElementSet::SearchDuplicateSurfaceId(const TSurfaceId& aSurfaceId)
  1.1441 +	{
  1.1442 +	TInt windowElements = iElements.Count();
  1.1443 +	for (TInt ii = 0; ii < windowElements; ii++)
  1.1444 +		{
  1.1445 +		if (iElements[ii]->SearchDuplicateSurfaceId(aSurfaceId))
  1.1446 +			{
  1.1447 +			return ETrue;
  1.1448 +			}
  1.1449 +		}
  1.1450 +	return EFalse;
  1.1451 +	}
  1.1452 +
  1.1453 +/** For a window, sets the opacity for the background surface and if setting to 
  1.1454 +0 removes placed surfaces.
  1.1455 +
  1.1456 +@param aWindow The window to perform the function on.
  1.1457 +@param aOpacity The opacity to set.
  1.1458 +@see CWindowElement::SetElementOpacity
  1.1459 +*/
  1.1460 +void CWindowElementSet::SetElementOpacity(CWsClientWindow& aWindow, TInt aOpacity)
  1.1461 +	{
  1.1462 +	WS_ASSERT_DEBUG(aOpacity == 0 || aOpacity == 255,EWsPanicUnexpectedOpacity);
  1.1463 +	return FindElement(aWindow)->SetElementOpacity(aOpacity, iScene);
  1.1464 +	}
  1.1465 +
  1.1466 +/** Finds all elements within the window that overlap with the region and 
  1.1467 +marks them as unassigned.
  1.1468 +If any part of these unassigned elements are outside the region, they are also 
  1.1469 +marked overlapping and locked config.
  1.1470 +
  1.1471 +@param aRedrawRegion The region in which elements will be unassigned
  1.1472 +@param aWindow The window to perform the function on.
  1.1473 +@return KErrNone on success or KErrNoMemory if critical calculations could not be made.
  1.1474 +@see CWindowElement::UnassignPlacedElements
  1.1475 +*/
  1.1476 +TInt CWindowElementSet::UnassignPlacedElements(const TRegion& aRedrawRegion, const CWsClientWindow& aWindow, TInt aGcDrawingCount)
  1.1477 +	{
  1.1478 +	return FindElement(aWindow)->UnassignPlacedElements(aRedrawRegion, aGcDrawingCount);
  1.1479 +	}
  1.1480 +
  1.1481 +/** Called to place a element within the window.
  1.1482 +It will either create a new element, recycle a element, or extend a element .
  1.1483 +
  1.1484 +@param aPlacedAttributes Returns the attributes of the surface placed.
  1.1485 +@param aSurfaceConfiguration The surface configuration for the surface to place.
  1.1486 +@param aUserDefinedRegion The user defined clipping of the window.
  1.1487 +@param aWindow The window to perform the function on.
  1.1488 +@return KErrNone on success or a system-wide error code.
  1.1489 +@see CWindowElement::AssignPlacedElement
  1.1490 +*/
  1.1491 +TInt CWindowElementSet::AssignPlacedElement(
  1.1492 +		TPlacedAttributes*& aPlacedAttributes,
  1.1493 +		const TSurfaceConfiguration& aSurfaceConfiguration,
  1.1494 +		const TRegion& aUserDefinedRegion,
  1.1495 +		const CWsClientWindow& aWindow,
  1.1496 +		TInt aGcDrawingCount	)
  1.1497 +	{
  1.1498 +	TInt index;
  1.1499 +	TInt error = KErrNone;
  1.1500 +	TInt result = FindEntry(aWindow, index);
  1.1501 +	TBool insertedElement = EFalse;
  1.1502 +	if (result == KErrNotFound)
  1.1503 +		{
  1.1504 +		CWindowElement* winelement = new CWindowElement(aWindow);
  1.1505 +		if (!winelement)
  1.1506 +			{
  1.1507 +			return KErrNoMemory;		//memory error
  1.1508 +			}
  1.1509 +		error = iElements.Insert(winelement, index);
  1.1510 +		if (error == KErrNone)
  1.1511 +			{
  1.1512 +			insertedElement = ETrue;
  1.1513 +			aWindow.Redraw()->SetHasElement(ETrue);
  1.1514 +			STACK_REGION windowRegion = aWindow.Abs();
  1.1515 +			if (!windowRegion.CheckError())
  1.1516 +				{
  1.1517 +				error = iElements[index]->UnassignPlacedElements(windowRegion, aGcDrawingCount);
  1.1518 +				}
  1.1519 +			else
  1.1520 +				{
  1.1521 +				error = KErrNoMemory;
  1.1522 +				}
  1.1523 +			windowRegion.Close();
  1.1524 +			}
  1.1525 +		else
  1.1526 +			{
  1.1527 +			delete winelement;
  1.1528 +			}
  1.1529 +		}
  1.1530 +	if (error == KErrNone)
  1.1531 +		{
  1.1532 +		
  1.1533 +		TInt assignReturn = iElements[index]->AssignPlacedElement(aPlacedAttributes, aSurfaceConfiguration,
  1.1534 +				aUserDefinedRegion, iScene, aGcDrawingCount);
  1.1535 +		error = assignReturn;	//return assign flags
  1.1536 +		}
  1.1537 +	if (error < KErrNone && insertedElement)
  1.1538 +		{	//remove this element that has just been created
  1.1539 +		aWindow.Redraw()->SetHasElement(EFalse);
  1.1540 +		iElements.Remove(index);
  1.1541 +		}
  1.1542 +	return error;
  1.1543 +	}
  1.1544 +
  1.1545 +/** Marks all elements that have been assigned since the unassign as drawn over.
  1.1546 +
  1.1547 +@param aWindow The window to perform the function on.
  1.1548 +@see CWindowElement::FlagAssignedElementsDrawnOver
  1.1549 +**/
  1.1550 +//void CWindowElementSet::FlagAssignedElementsDrawnOver(const CWsClientWindow& aWindow)
  1.1551 +//	{
  1.1552 +//	FindElement(aWindow)->FlagAssignedElementsDrawnOver();
  1.1553 +//	}
  1.1554 +
  1.1555 +/** Called at the end of a redraw to set cached changed to elements, and
  1.1556 +removed elements that are still unassigned.
  1.1557 +
  1.1558 +@param aWindow The window to perform the function on.
  1.1559 +@return Flags to describe what extra work needed doing.
  1.1560 +@see CWindowElement::CleanUpPlacedElements
  1.1561 +*/
  1.1562 +TInt CWindowElementSet::CleanUpPlacedElements(const CWsClientWindow& aWindow, TInt aGcDrawingCount)
  1.1563 +	{
  1.1564 +	TInt index;
  1.1565 +	TInt error = FindEntry(aWindow,index);
  1.1566 +	if (error < 0)
  1.1567 +		{
  1.1568 +		WS_ASSERT_DEBUG(0,EWsPanicNoWindowElement);
  1.1569 +		return error;
  1.1570 +		}
  1.1571 +	TInt returnFlags = iElements[index]->CleanUpPlacedElements(iScene, aGcDrawingCount);
  1.1572 +
  1.1573 +	if (!iElements[index]->iBackgroundElement.iElement && iElements[index]->iPlacedElements.Count() == 0)
  1.1574 +		{	//remove window entry
  1.1575 +		DestroyWindowElementEntry(index);
  1.1576 +		}
  1.1577 +	else
  1.1578 +		{
  1.1579 +		if (returnFlags&CWindowElement::EResort)
  1.1580 +			{	//need to sort elements
  1.1581 +			//find front most visible element in window behind, invoke sort by z order
  1.1582 +			if (index == 0)
  1.1583 +				{	//this element is backmost window
  1.1584 +				iElements[index]->MoveToAboveGivenElement(iScene,NULL);
  1.1585 +				}
  1.1586 +			else
  1.1587 +				{	//place infront of highest element behind
  1.1588 +				if (iElements[index-1]->iPlacedElements.Count() == 0)
  1.1589 +					{	//top element of element behind must be the background element
  1.1590 +					iElements[index]->MoveToAboveGivenElement(iScene,iElements[index-1]->iBackgroundElement.iElement);				
  1.1591 +					}
  1.1592 +				else
  1.1593 +					{	//top element of element behind must be highest placed element
  1.1594 +					TInt placedCount = iElements[index-1]->iPlacedElements.Count();
  1.1595 +					iElements[index]->MoveToAboveGivenElement(iScene,iElements[index-1]->iPlacedElements[placedCount-1].iElement);
  1.1596 +					}
  1.1597 +				}
  1.1598 +			}
  1.1599 +		}
  1.1600 +	if (returnFlags&CWindowElement::EFailed)
  1.1601 +		{	//visible elements may have been deleted, error to suggest a full window redraw
  1.1602 +		return KErrGeneral;
  1.1603 +		}
  1.1604 +	return returnFlags;
  1.1605 +	}
  1.1606 +
  1.1607 +/** Counts how many elements there are in the set.
  1.1608 +
  1.1609 +@return The number of elements in the set.
  1.1610 +*/
  1.1611 +TInt CWindowElementSet::Count() const
  1.1612 +	{
  1.1613 +	return iElements.Count();
  1.1614 +	}
  1.1615 +
  1.1616 +/** Returns the background attributes for the specified window
  1.1617 +
  1.1618 +@param aWindow The window to perform the function on.
  1.1619 +@return The background surface attributes.  If the window has no elementset entry, returns NULL.
  1.1620 +*/
  1.1621 +TBackgroundAttributes* CWindowElementSet::FindBackgroundElement(const CWsClientWindow& aWindow)
  1.1622 +	{
  1.1623 +	TInt index;
  1.1624 +	TInt err = FindEntry(aWindow, index);
  1.1625 +
  1.1626 +	if (err != KErrNotFound)
  1.1627 +		{
  1.1628 +		return &(iElements[index]->iBackgroundElement);
  1.1629 +		}
  1.1630 +	return NULL;
  1.1631 +	}
  1.1632 +
  1.1633 +/**	Returns the contents of the element data associated with the input window.
  1.1634 +If this method is successful, then neither pointer will be NULL.
  1.1635 +
  1.1636 +@return standard symbian error code.
  1.1637 +@param	aWindow	window to find
  1.1638 +@param	aBackAttr	backgroud surface attributes associated with the window
  1.1639 +@param	aPlacedAttr	array of placed surface attributes associated with the window.
  1.1640 +*/
  1.1641 +TInt CWindowElementSet::FindElements(	CWsClientWindow const &aWindow, 
  1.1642 +									TBackgroundAttributes const * & aBackAttr, 
  1.1643 +									RArray<class TPlacedAttributes> const * &	aPlacedAttr	)
  1.1644 +	{
  1.1645 +	TInt index;
  1.1646 +
  1.1647 +	TInt err = FindEntry(aWindow, index);
  1.1648 +//	__ASSERT_DEBUG(err != KErrNotFound, Panic(EWsPanicNoWindowElement));
  1.1649 +
  1.1650 +//	return *iElements[index].iElement;
  1.1651 +	if (err >= KErrNone)
  1.1652 +		{
  1.1653 +		aBackAttr=&iElements[index]->iBackgroundElement;
  1.1654 +		aPlacedAttr=&iElements[index]->iPlacedElements;
  1.1655 +		}
  1.1656 +	else
  1.1657 +		{
  1.1658 +		aBackAttr=NULL;
  1.1659 +		aPlacedAttr=NULL;
  1.1660 +		}
  1.1661 +	return err;
  1.1662 +	
  1.1663 +	}
  1.1664 +
  1.1665 +/** Registers the surface with the compositor.
  1.1666 +@param aSurface The surface id to register.
  1.1667 +@return  KErrNone if successful, KErrNoMemory if registration fails due to low
  1.1668 +memory, KErrNotSupported if the surface is not compatible with
  1.1669 +this compositor or KErrBadHandle if the given surface ID does not
  1.1670 +represent a valid surface. KErrArgument is returned if the
  1.1671 +surface does not have both dimensions less than 32 767 pixels.
  1.1672 +*/
  1.1673 +TInt CWindowElementSet::RegisterSurface(const TSurfaceId& aSurface)
  1.1674 +    {
  1.1675 +    
  1.1676 +    return iScene.RegisterSurface(aSurface);
  1.1677 +    }
  1.1678 +
  1.1679 +/** Unregisters the surface with the compositor.
  1.1680 +
  1.1681 +@param aSurface The surface id to register.
  1.1682 +@return  KErrNone if successful. KErrInUse if the surface is
  1.1683 +used by a layer or layers. KErrBadHandle if the surface
  1.1684 +is not currently registered. KErrArgument if
  1.1685 +the surface ID is a NULL ID.
  1.1686 +*/
  1.1687 +
  1.1688 +void CWindowElementSet::UnregisterSurface(const TSurfaceId& aSurface)
  1.1689 +    {
  1.1690 +    
  1.1691 +    TInt returnCode = iScene.UnregisterSurface(aSurface);
  1.1692 +    __ASSERT_DEBUG((returnCode==KErrNone || returnCode==KErrInUse || returnCode==KErrBadHandle), Panic(EWsPanicSceneErrorIgnored));
  1.1693 +    }
  1.1694 +
  1.1695 +/** Sorts the array elements into the same order as the windows are in the
  1.1696 +hierarchy. Use after window hierarchy has been modified, to update the scene
  1.1697 +order to match.
  1.1698 +@return EFalse if element order unchanged, ETrue if order may have changed.
  1.1699 +*/
  1.1700 +TBool CWindowElementSet::SortByZOrder()
  1.1701 +	{
  1.1702 +	if (iElements.Count() < 2)
  1.1703 +		{
  1.1704 +		// Early out for the very common cases where there are zero or one
  1.1705 +		// elements, which cannot therefore be out of order.
  1.1706 +		return EFalse;
  1.1707 +		}
  1.1708 +
  1.1709 +	// The approach being used is to first just sort the array, then update the 
  1.1710 +	// scene order afterwards. This is simple to code and since there are not
  1.1711 +	// expected to be many elements, it should be perfectly good enough.
  1.1712 +
  1.1713 +	TLinearOrder<CWindowElement> order(WindowOrder);
  1.1714 +	iElements.Sort(order);
  1.1715 +
  1.1716 +	TBool orderChanged = EFalse;
  1.1717 +	MWsElement* below;
  1.1718 +	TInt elementCount = iElements.Count();
  1.1719 +	for (TInt index = 0; index < elementCount; index++)
  1.1720 +		{
  1.1721 +		if (index == 0)
  1.1722 +			{
  1.1723 +			below = NULL;
  1.1724 +			}
  1.1725 +		else
  1.1726 +			{
  1.1727 +			//Find highest element in window below
  1.1728 +			TInt placedCount = iElements[index-1]->iPlacedElements.Count();
  1.1729 +			if (placedCount > 0)
  1.1730 +				{
  1.1731 +				below = iElements[index-1]->iPlacedElements[placedCount-1].iElement;
  1.1732 +				}
  1.1733 +			else
  1.1734 +				{	//below = above = background surface
  1.1735 +				below = iElements[index-1]->iBackgroundElement.iElement;
  1.1736 +				}
  1.1737 +			}
  1.1738 +		//Find lowest element in above window element
  1.1739 +		MWsElement* above;
  1.1740 +		if (iElements[index]->iBackgroundElement.iElement != NULL)
  1.1741 +			{	//use background element
  1.1742 +			above = iElements[index]->iBackgroundElement.iElement;	//background element will be bottom
  1.1743 +			}
  1.1744 +		else	//use bottom element of placed surfaces
  1.1745 +			{
  1.1746 +			//above = iElements[index]->iPlacedElements[iElements[index]->iPlacedElements.Count()-1].iElement;	//first or last placed element is bottom?
  1.1747 +			above = iElements[index]->iPlacedElements[0].iElement;
  1.1748 +			}
  1.1749 +		if (above->ElementBelow() != below)
  1.1750 +			{
  1.1751 +			//CALL below window element function to move all elements above 'below'
  1.1752 +			iElements[index]->MoveToAboveGivenElement(iScene, below);
  1.1753 +	        orderChanged = ETrue;
  1.1754 +			}
  1.1755 +		}
  1.1756 +	return orderChanged;
  1.1757 +	}
  1.1758 +
  1.1759 +/** Processes the specified windows placed elements, for when windows visibility is changed.
  1.1760 +If the placed elements are no longer visible, they are removed.
  1.1761 +
  1.1762 +@param aWindow The window to call the function on.
  1.1763 +@return Positive if any elements have been removed, zero if not, or errorcode. 
  1.1764 +@see CWindowElement::SetVisibleRegion
  1.1765 +*/
  1.1766 +TInt CWindowElementSet::SetVisibleRegion(CWsClientWindow& aWindow)
  1.1767 +	{
  1.1768 +	TInt index;
  1.1769 +	TInt find = FindEntry(aWindow,index);
  1.1770 +	WS_ASSERT_DEBUG(find>=KErrNone,EWsPanicNoWindowElement);
  1.1771 +	TBool ret = iElements[index]->SetVisibleRegion(aWindow.VisibleRegion(), iScene);
  1.1772 +
  1.1773 +	if (!iElements[index]->iBackgroundElement.iElement && iElements[index]->iPlacedElements.Count() == 0)
  1.1774 +		{
  1.1775 +		DestroyWindowElementEntry(index);
  1.1776 +		}
  1.1777 +	return ret;
  1.1778 +	}
  1.1779 +
  1.1780 +/** Updates the specified windows elements extent, either from a window movement or a change in window size
  1.1781 +
  1.1782 +@param aWindow The window to call the function on.
  1.1783 +@param aOffset The movement of the window.  If NULL then the window has changed size.
  1.1784 +@see CWindowElement::UpdateElementExtent
  1.1785 +*/
  1.1786 +void CWindowElementSet::UpdateElementExtent(const CWsClientWindow& aWindow, const TPoint* aOffset)
  1.1787 +	{
  1.1788 +	FindElement(aWindow)->UpdateElementExtent(aOffset);
  1.1789 +	}
  1.1790 +
  1.1791 +/** Method to fill in a TSurfaceConfiguration from a scene element.
  1.1792 +
  1.1793 +@param aConfiguration Surface configuration to fill in.
  1.1794 +@param aElement Element to get information from.
  1.1795 +@return Once multiple versions of TSurfaceConfiguration are available, KErrNotSupported if configuration supplied is too small.
  1.1796 +*/
  1.1797 +TInt CWindowElementSet::GetConfiguration(TSurfaceConfiguration& aConfiguration,MWsElement& aElement)
  1.1798 +	{
  1.1799 +	TSurfaceId tempSurfaceId = aElement.ConnectedSurface();
  1.1800 +	aConfiguration.SetSurfaceId(tempSurfaceId);
  1.1801 +	
  1.1802 +	//Convert and copy orientation
  1.1803 +	aConfiguration.SetOrientation(ElementToGcRotation(aElement.SourceRotation()));
  1.1804 +	
  1.1805 +	//Convert and copy flip
  1.1806 +	TBool flip = aElement.SourceFlipping();
  1.1807 +	aConfiguration.SetFlip(flip);
  1.1808 +
  1.1809 +	//Convert and copy viewport
  1.1810 +	TRect tempViewport;
  1.1811 +	aElement.GetSourceRectangle(tempViewport);
  1.1812 +	aConfiguration.SetViewport(tempViewport);
  1.1813 +	TRect tempExtent;
  1.1814 +	aElement.GetDestinationRectangle(tempExtent);
  1.1815 +	aConfiguration.SetExtent(tempExtent);
  1.1816 +	return KErrNone;	//Could fail if there are multiple versions of TSurfaceConfiguration
  1.1817 +	}
  1.1818 +
  1.1819 +/** Returns the window element entry for the specified window.
  1.1820 +
  1.1821 +@param aWindow The window to call the function on.
  1.1822 +*/
  1.1823 +CWindowElement* CWindowElementSet::FindElement(const CWsClientWindow& aWindow) const
  1.1824 +	{
  1.1825 +	TInt index;
  1.1826 +	TInt error = FindEntry(aWindow,index);
  1.1827 +	if (error == KErrNone)
  1.1828 +		return iElements[index];
  1.1829 +	WS_ASSERT_DEBUG(EFalse,EWsPanicNoWindowElement);
  1.1830 +	return NULL;
  1.1831 +	}
  1.1832 +
  1.1833 +/** Creates a set of window-element pairs for this scene.
  1.1834 +
  1.1835 +@param aScene To allow access to the scene.
  1.1836 +@param aComposer To allow access to the composer.
  1.1837 +*/
  1.1838 +CWindowElementSet::CWindowElementSet(MWsScene& aScene) :
  1.1839 +	iScene(aScene)
  1.1840 +	{}
  1.1841 +
  1.1842 +/** Searches for the entry in iElements with the given window
  1.1843 +
  1.1844 +@param aWindow The window to find the entry of / where it should be inserted.
  1.1845 +@param aIndex aIndex is set to entry found or the insertion point, respectively.
  1.1846 +@param aLinearSearch ETrue if a linear search of the window element set is required.
  1.1847 +@return KErrNone if found or KErrNotFound.
  1.1848 +*/
  1.1849 +TInt CWindowElementSet::FindEntry(const CWsClientWindow& aWindow, TInt& aIndex, TBool aLinearSearch /*= EFalse*/) const
  1.1850 +	{
  1.1851 +	CWindowElement winelement(aWindow);
  1.1852 +	
  1.1853 +	// The array order makes use of the parent pointer, which gets reset during
  1.1854 +	// window shutdown, so if it is clear fall back to a linear search.
  1.1855 +	if (!aWindow.BaseParent() || aLinearSearch)
  1.1856 +		{
  1.1857 +		TIdentityRelation<CWindowElement> match(WindowMatch);
  1.1858 +		
  1.1859 +		aIndex = iElements.Find(&winelement, match);
  1.1860 +		return (aIndex == KErrNotFound) ? KErrNotFound : KErrNone;
  1.1861 +		}
  1.1862 +	else
  1.1863 +		{
  1.1864 +		TLinearOrder<CWindowElement> order(WindowOrder);
  1.1865 +		return iElements.FindInOrder(&winelement, aIndex, order);
  1.1866 +		}
  1.1867 +	}
  1.1868 +
  1.1869 +/** Used to find an entry in the set when order cannot be used. 
  1.1870 +
  1.1871 +@param aFirst First windowelement to compare.
  1.1872 +@param aSecond Second windowelement to compare.
  1.1873 +@return ETrue if the entries are the same, and EFalse if they are not.
  1.1874 +*/
  1.1875 +TBool CWindowElementSet::WindowMatch(const CWindowElement& aFirst, const CWindowElement& aSecond)
  1.1876 +	{
  1.1877 +	return (&aFirst.iWindow == &aSecond.iWindow);
  1.1878 +	}
  1.1879 +
  1.1880 +
  1.1881 +/** Used to determine the order of entries in the set.
  1.1882 +
  1.1883 +@param aFirst First windowelement to compare.
  1.1884 +@param aSecond Second windowelement to compare.
  1.1885 +@return zero if the entries are the same, a negative value if 
  1.1886 +aFirst is behind aSecond or a positive value if aFirst is in front of aSecond.
  1.1887 +*/
  1.1888 +TInt CWindowElementSet::WindowOrder(const CWindowElement& aFirst, const CWindowElement& aSecond)
  1.1889 +	{
  1.1890 +	TInt result = 0;
  1.1891 +
  1.1892 +	if (&aFirst.iWindow != &aSecond.iWindow)
  1.1893 +		{
  1.1894 +		result = aFirst.iWindow.IsInfrontOf(&aSecond.iWindow) ? 1 : -1;
  1.1895 +		}
  1.1896 +
  1.1897 +	return result;
  1.1898 +	}
  1.1899 +
  1.1900 +
  1.1901 +/**	Re-sends the extents for all the elements (in this window) to the scene  	
  1.1902 + * 	This allows the renderstage to re-scale those element extents
  1.1903 + * 
  1.1904 + * 
  1.1905 + **/
  1.1906 +void CWindowElement::ResubmitAllElementExtents()
  1.1907 +	{
  1.1908 +	if (MWsElement* element=iBackgroundElement.iElement)
  1.1909 +		{
  1.1910 +		TRect extent(TRect::EUninitialized);
  1.1911 +		element->GetDestinationRectangle(extent);
  1.1912 +		element->SetDestinationRectangle(extent);
  1.1913 +		}
  1.1914 +	TInt elementCount = iPlacedElements.Count();
  1.1915 +	for (TInt index = 0; index < elementCount; index++)
  1.1916 +		{
  1.1917 +		if (MWsElement* element=iPlacedElements[index].iElement)
  1.1918 +			{
  1.1919 +			TRect extent(TRect::EUninitialized);
  1.1920 +			element->GetDestinationRectangle(extent);
  1.1921 +			element->SetDestinationRectangle(extent);
  1.1922 +			}
  1.1923 +		}
  1.1924 +	}
  1.1925 +
  1.1926 +/**	Re-sends the extents for all the elements (in all the windows) to the scene  	
  1.1927 + * 	This allows the renderstage to re-scale those element extents
  1.1928 + * 
  1.1929 + * 
  1.1930 + **/
  1.1931 +void CWindowElementSet::ResubmitAllElementExtents()
  1.1932 +	{
  1.1933 +	TInt elementCount = iElements.Count();
  1.1934 +	for (TInt index = 0; index < elementCount; index++)
  1.1935 +		{
  1.1936 +		iElements[index]->ResubmitAllElementExtents();
  1.1937 +		}
  1.1938 +	}
  1.1939 +
  1.1940 +//
  1.1941 +// Debug functions
  1.1942 +//
  1.1943 +
  1.1944 +/** Returns background attributes for the specified window index.
  1.1945 +For use with debug client interface.
  1.1946 +
  1.1947 +@param aWin Window index to get the surface attributes from.
  1.1948 +@return Background attributes for the specified window.  NULL if the window index is invalid.
  1.1949 +*/
  1.1950 +const TBackgroundAttributes* CWindowElementSet::DebugBackgroundAt(TUint aWin)const
  1.1951 +	{
  1.1952 +	if (aWin>=Count())
  1.1953 +		return NULL;
  1.1954 +	return &iElements[aWin]->iBackgroundElement;
  1.1955 +	}
  1.1956 +
  1.1957 +/** Returns the client window for the specified window index.
  1.1958 +For use with debug client interface.
  1.1959 +
  1.1960 +@param aWin Window index to get the client window from.
  1.1961 +@return Client window for the specified window.  NULL if the window index is invalid.
  1.1962 +*/
  1.1963 +const CWsClientWindow* CWindowElementSet::DebugWindowAt(TUint aWin)const
  1.1964 +	{
  1.1965 +	if (aWin>=Count())
  1.1966 +		return NULL;
  1.1967 +	return &iElements[aWin]->iWindow;
  1.1968 +	}
  1.1969 +
  1.1970 +/** Returns how many placed surfaces are on the specified window
  1.1971 +For use with debug client interface.
  1.1972 +
  1.1973 +@param aWin Window index to get the client window from.
  1.1974 +@return Amount of placed surfaces for the specified window index
  1.1975 +*/
  1.1976 +TInt CWindowElementSet::DebugPlacedCountAt(TUint aWin)const
  1.1977 +	{
  1.1978 +	if (aWin>=Count())
  1.1979 +		return -1;
  1.1980 +	return iElements[aWin]->iPlacedElements.Count();
  1.1981 +	}
  1.1982 +
  1.1983 +/** Returns the placed attributes for the specified placed surface index in the
  1.1984 +specified window index.
  1.1985 +For use with debug client interface.
  1.1986 +
  1.1987 +@param aWin Window index to get the client window from.
  1.1988 +@param aPlace Placed surface index to get the placed surface attributes
  1.1989 +@return Placed surface attributes.  NULL if either index was invalid.
  1.1990 +*/
  1.1991 +const TPlacedAttributes* CWindowElementSet::DebugPlacedAt(TUint aWin,TUint aPlace)const
  1.1992 +	{
  1.1993 +	if (aWin>=Count())
  1.1994 +		return NULL;
  1.1995 +	if (aPlace>=iElements[aWin]->iPlacedElements.Count())
  1.1996 +		return NULL;
  1.1997 +	return &iElements[aWin]->iPlacedElements[aPlace];
  1.1998 +	}
  1.1999 +
  1.2000 +MWsElement* CWindowElementSet::GetElementFromWindow(const CWsClientWindow& aWindow) const
  1.2001 +    {
  1.2002 +    CWindowElement* windowElement = FindElement(aWindow);
  1.2003 +    if (windowElement)
  1.2004 +        return windowElement->Element();
  1.2005 +    else 
  1.2006 +        return NULL; 
  1.2007 +    }