os/graphics/windowing/windowserver/nga/SERVER/openwfc/windowelementset.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#include <e32std.h>
sl@0
    17
#include "cliwin.h"
sl@0
    18
#include "server.h"
sl@0
    19
#include "windowelementset.h"
sl@0
    20
#include "regionextend.h"
sl@0
    21
#include <graphics/wsscene.h>
sl@0
    22
#include <graphics/wsgraphicdrawer.h>
sl@0
    23
#include "panics.h"
sl@0
    24
#include "wstop.h"
sl@0
    25
sl@0
    26
#include "EVENT.H"
sl@0
    27
sl@0
    28
#ifndef KCacheListGranularity
sl@0
    29
	enum	{KCacheListGranularity = 8};
sl@0
    30
#endif
sl@0
    31
#ifndef KCacheExtensionReserve
sl@0
    32
	enum	{KCacheExtensionReserve = 1};
sl@0
    33
#endif
sl@0
    34
	
sl@0
    35
sl@0
    36
sl@0
    37
/** Element to gc rotation function
sl@0
    38
*/
sl@0
    39
LOCAL_C inline CFbsBitGc::TGraphicsOrientation ElementToGcRotation(MWsElement::TElementRotation aElementRotation)
sl@0
    40
	{
sl@0
    41
	CFbsBitGc::TGraphicsOrientation gcRotation = CFbsBitGc::EGraphicsOrientationNormal;
sl@0
    42
sl@0
    43
	switch (aElementRotation)
sl@0
    44
		{
sl@0
    45
		case MWsElement::EElementAntiClockwise90:
sl@0
    46
			gcRotation = CFbsBitGc::EGraphicsOrientationRotated90;
sl@0
    47
			break;
sl@0
    48
		case MWsElement::EElementAntiClockwise180:
sl@0
    49
			gcRotation = CFbsBitGc::EGraphicsOrientationRotated180;
sl@0
    50
			break;
sl@0
    51
		case MWsElement::EElementAntiClockwise270:
sl@0
    52
			gcRotation = CFbsBitGc::EGraphicsOrientationRotated270;
sl@0
    53
			break;
sl@0
    54
		default:
sl@0
    55
			break;
sl@0
    56
		}
sl@0
    57
	return gcRotation;
sl@0
    58
	}
sl@0
    59
sl@0
    60
//
sl@0
    61
// CWindowElement
sl@0
    62
//
sl@0
    63
sl@0
    64
CWindowElement::CWindowElement(const CWsClientWindow& aWindow) :
sl@0
    65
	iWindow(aWindow), iCache(KCacheListGranularity), iHighestReusedIndex(-1)
sl@0
    66
	{}
sl@0
    67
sl@0
    68
sl@0
    69
/** Finds all elements within the window that overlap with the region and 
sl@0
    70
marks them as unassigned.
sl@0
    71
If any part of these unassigned elements are outside the region, they are also 
sl@0
    72
marked overlapping and locked config.
sl@0
    73
sl@0
    74
@param aRegion The region that is being redrawn.
sl@0
    75
@return KErrNone on success or KErrNoMemory if critical calculations could not be made.
sl@0
    76
sl@0
    77
@see CWindowElementSet::UnassignPlacedElements
sl@0
    78
**/
sl@0
    79
TInt CWindowElement::UnassignPlacedElements(const TRegion& aRegion,TInt aGcDrawingCount)
sl@0
    80
	{
sl@0
    81
	iFlags = 0;	//reset flags
sl@0
    82
	iStartGcDrawingCount = aGcDrawingCount;	//Detect any drawing between calls
sl@0
    83
	iBackgroundElement.SetDrawnOverLast(iBackgroundElement.DrawnOver());
sl@0
    84
	iNonRedrawVisibleRegion.Clear();
sl@0
    85
	iNonRedrawVisibleRegion.Copy(const_cast<CWsClientWindow&>(iWindow).VisibleRegion());
sl@0
    86
	iNonRedrawVisibleRegion.SubRegion(aRegion);
sl@0
    87
	if (iNonRedrawVisibleRegion.CheckError())
sl@0
    88
		{	//memory error. Still needs to cope if whole window is being redrawn
sl@0
    89
		iNonRedrawVisibleRegion.Clear();
sl@0
    90
		TInt fullWindowRedraw = TRegionExtend::Cast(aRegion).TestDifference(const_cast<CWsClientWindow&>(iWindow).VisibleRegion());
sl@0
    91
		if (fullWindowRedraw != 0)
sl@0
    92
			{	//not a full window redraw, return immediately
sl@0
    93
			return KErrNoMemory;
sl@0
    94
			}
sl@0
    95
		}	
sl@0
    96
	TInt placedElements = iPlacedElements.Count();
sl@0
    97
	TInt unassignedCount = 0;
sl@0
    98
	TInt ii;
sl@0
    99
	TBool cacheError = EFalse;
sl@0
   100
sl@0
   101
	STACK_REGION tempSurfaceBaseArea;
sl@0
   102
	if (iCache.Reserve(placedElements + KCacheExtensionReserve) < KErrNone)	//quite often there may need to be room for more
sl@0
   103
		{	//failed to reserve space
sl@0
   104
		cacheError = ETrue;
sl@0
   105
		}
sl@0
   106
	TRect tempExtent;
sl@0
   107
	TRect tempIntersect;
sl@0
   108
	
sl@0
   109
	for (ii=0; ii<placedElements; ii++)
sl@0
   110
		{
sl@0
   111
		if (!cacheError)
sl@0
   112
			{	//cache is ok up to this point but does not contain cache for this element yet
sl@0
   113
			RRegion newRegion;
sl@0
   114
			if (iCache.Append(newRegion) < KErrNone)
sl@0
   115
				{
sl@0
   116
				cacheError = ETrue;
sl@0
   117
				}
sl@0
   118
			newRegion.Close();
sl@0
   119
			}
sl@0
   120
sl@0
   121
		iPlacedElements[ii].iElement->GetDestinationRectangle(tempExtent);
sl@0
   122
		TInt regionReturn = TRegionExtend::Cast(aRegion).TestDifference(tempExtent);
sl@0
   123
		iPlacedElements[ii].SetDrawnOverLast(iPlacedElements[ii].DrawnOver());	//allows detection of change of state
sl@0
   124
		if (!(regionReturn&TRegionExtend::ENoIntersect))
sl@0
   125
			{	//redraw region intersects with element
sl@0
   126
			iPlacedElements[ii].SetUnassigned();
sl@0
   127
			iPlacedElements[ii].SetDrawnOver(EFalse);
sl@0
   128
			unassignedCount++;
sl@0
   129
			
sl@0
   130
			if (!(regionReturn&TRegionExtend::EAdd))
sl@0
   131
				{	//surface is entirely within the redraw region
sl@0
   132
				iPlacedElements[ii].SetOverlapping(EFalse);
sl@0
   133
				iPlacedElements[ii].SetLockedConfig(EFalse);
sl@0
   134
				if(!cacheError)
sl@0
   135
					{
sl@0
   136
					TRect destinationRect;
sl@0
   137
					tempSurfaceBaseArea.Clear();
sl@0
   138
                    iPlacedElements[ii].iElement->GetDestinationRectangle(destinationRect);
sl@0
   139
					tempSurfaceBaseArea.AddRect(destinationRect);
sl@0
   140
					if (!tempSurfaceBaseArea.CheckError())
sl@0
   141
						{
sl@0
   142
						tempSurfaceBaseArea.Offset(tempExtent.iTl);	//cache needs to be in absolute coords
sl@0
   143
						iCache[ii].Copy(tempSurfaceBaseArea);
sl@0
   144
						}
sl@0
   145
					if (tempSurfaceBaseArea.CheckError() || iCache[ii].CheckError())
sl@0
   146
						{	//cache is no good from this point onwards
sl@0
   147
						iCache[ii].Close();
sl@0
   148
						iCache.Remove(ii);
sl@0
   149
						cacheError = ETrue;
sl@0
   150
						}
sl@0
   151
					}
sl@0
   152
				}
sl@0
   153
			else
sl@0
   154
				{
sl@0
   155
                TRect destinationRect;
sl@0
   156
                tempSurfaceBaseArea.Clear();
sl@0
   157
                iPlacedElements[ii].iElement->GetDestinationRectangle(destinationRect);
sl@0
   158
                tempSurfaceBaseArea.AddRect(destinationRect);
sl@0
   159
				if (tempSurfaceBaseArea.CheckError())
sl@0
   160
					{	//critical failure, clear cache and return error
sl@0
   161
					for (ii=iCache.Count(); ii>0; ii--)
sl@0
   162
						{
sl@0
   163
						iCache[ii-1].Close();
sl@0
   164
						iCache.Remove(ii-1);
sl@0
   165
						}
sl@0
   166
					iCache.Reset();
sl@0
   167
					iNonRedrawVisibleRegion.Close();
sl@0
   168
					tempSurfaceBaseArea.Close();
sl@0
   169
					return KErrNoMemory;
sl@0
   170
					}
sl@0
   171
				else
sl@0
   172
					{
sl@0
   173
					tempSurfaceBaseArea.Offset(tempExtent.iTl);	//change to absolute coordinates
sl@0
   174
					regionReturn = TRegionExtend::Cast(iNonRedrawVisibleRegion ).TestDifference(tempSurfaceBaseArea);
sl@0
   175
					if (regionReturn&TRegionExtend::ENoIntersect)
sl@0
   176
						{	//element base area entirely inside redraw region
sl@0
   177
						iPlacedElements[ii].SetOverlapping(EFalse);
sl@0
   178
						iPlacedElements[ii].SetLockedConfig(EFalse);
sl@0
   179
						}
sl@0
   180
					else
sl@0
   181
						{	//element base area is at least partly within non redraw visible region
sl@0
   182
						iPlacedElements[ii].SetOverlapping(ETrue);
sl@0
   183
						iPlacedElements[ii].SetLockedConfig(ETrue);
sl@0
   184
						}
sl@0
   185
					if(!cacheError)
sl@0
   186
						{
sl@0
   187
						iCache[ii].Copy(tempSurfaceBaseArea);
sl@0
   188
						if (iCache[ii].CheckError())
sl@0
   189
							{	//cache is no good from this point onwards
sl@0
   190
							iCache[ii].Close();
sl@0
   191
							iCache.Remove(ii);
sl@0
   192
							cacheError = ETrue;
sl@0
   193
							}
sl@0
   194
						}
sl@0
   195
					}
sl@0
   196
				}
sl@0
   197
			}
sl@0
   198
		else
sl@0
   199
			{	//element does not intersect with redraw region
sl@0
   200
			iPlacedElements[ii].SetUnassigned(EFalse);
sl@0
   201
			}
sl@0
   202
		}
sl@0
   203
	tempSurfaceBaseArea.Close();
sl@0
   204
	
sl@0
   205
	if (iRemovedSurfaces.Reserve(unassignedCount) == KErrNone)
sl@0
   206
		{
sl@0
   207
		iRemovedSurfacesValid = ETrue;
sl@0
   208
		}
sl@0
   209
	else
sl@0
   210
		{
sl@0
   211
		iRemovedSurfacesValid = EFalse;
sl@0
   212
		}
sl@0
   213
	return KErrNone;
sl@0
   214
	}
sl@0
   215
sl@0
   216
/** Marks all elements that have been assigned since the unassign as drawn over.
sl@0
   217
sl@0
   218
@see CWindowElementSet::FlagAssignedElementsDrawnOver
sl@0
   219
**/
sl@0
   220
void CWindowElement::FlagAssignedElementsDrawnOver(TInt aGcDrawingCount)
sl@0
   221
	{
sl@0
   222
	iStartGcDrawingCount = aGcDrawingCount;
sl@0
   223
	if (iBackgroundElement.iElement)
sl@0
   224
		{
sl@0
   225
		iBackgroundElement.SetDrawnOver();
sl@0
   226
		}
sl@0
   227
	TInt placedElementCount = iPlacedElements.Count();
sl@0
   228
	for (TInt ii = 0; ii < placedElementCount; ii++)
sl@0
   229
		{
sl@0
   230
		if(!iPlacedElements[ii].Unassigned() && iPlacedElements[ii].LockedConfig())
sl@0
   231
			{
sl@0
   232
			iPlacedElements[ii].SetDrawnOver();
sl@0
   233
			}
sl@0
   234
		else if(iPlacedElements[ii].Unassigned() && iPlacedElements[ii].LockedConfig() &&
sl@0
   235
				!iPlacedElements[ii].ChangedClip() && iPlacedElements[ii].Overlapping())
sl@0
   236
			{
sl@0
   237
			iPlacedElements[ii].SetDrawnOver();
sl@0
   238
			}
sl@0
   239
		}
sl@0
   240
	}
sl@0
   241
sl@0
   242
/** Checks whether a elements configuration is the same as that of the surface configuration, taking
sl@0
   243
into account the stateflags (TPlacedAttributes) passed in.
sl@0
   244
It must match surfaceId and extent.
sl@0
   245
If it is a locked config it will need to match viewport and orientation as well.
sl@0
   246
sl@0
   247
@param aElement Element to compare against.
sl@0
   248
@param aSurfaceConfiguration Configuration to compare.
sl@0
   249
@param aStateFlags The TPlacedAttributes flags for the element.
sl@0
   250
@return ETrue if a match, otherwise EFalse.
sl@0
   251
*/
sl@0
   252
TBool CWindowElement::IsElementAMatch(MWsElement& aElement, const TSurfaceConfiguration& aSurfaceConfiguration,
sl@0
   253
		TInt aStateFlags)
sl@0
   254
	{
sl@0
   255
	TBool match = EFalse;
sl@0
   256
	
sl@0
   257
	TRect tempElementExtent;
sl@0
   258
	TRect tempCRPExtent;
sl@0
   259
	aElement.GetDestinationRectangle(tempElementExtent);
sl@0
   260
	aSurfaceConfiguration.GetExtent(tempCRPExtent);
sl@0
   261
	if (tempElementExtent == tempCRPExtent)
sl@0
   262
		{
sl@0
   263
		TSurfaceId tempElementSurfaceId;
sl@0
   264
		TSurfaceId tempCRPSurfaceId;
sl@0
   265
		aSurfaceConfiguration.GetSurfaceId(tempElementSurfaceId);
sl@0
   266
		tempCRPSurfaceId = aElement.ConnectedSurface();
sl@0
   267
		if (tempElementSurfaceId == tempCRPSurfaceId)
sl@0
   268
			{
sl@0
   269
			match = ETrue;
sl@0
   270
			if (aStateFlags&TPlacedAttributes::ELockedConfig)
sl@0
   271
				{
sl@0
   272
				match = EFalse;
sl@0
   273
				TBool orientationOK = EFalse;
sl@0
   274
				TRect tempCRPViewport;
sl@0
   275
				aSurfaceConfiguration.GetViewport(tempCRPViewport);
sl@0
   276
				//if explicit viewport flagged
sl@0
   277
				TRect tempElementViewport;
sl@0
   278
				aElement.GetSourceRectangle(tempElementViewport);
sl@0
   279
				if (tempElementViewport == tempCRPViewport)
sl@0
   280
					{
sl@0
   281
					CFbsBitGc::TGraphicsOrientation tempElementOrientation;
sl@0
   282
					MWsElement::TElementRotation tempCRPRotation;
sl@0
   283
					CFbsBitGc::TGraphicsOrientation tempCRPOrientation;
sl@0
   284
					tempElementOrientation = aSurfaceConfiguration.Orientation();
sl@0
   285
					tempCRPRotation = aElement.SourceRotation();
sl@0
   286
					tempCRPOrientation = ElementToGcRotation(tempCRPRotation);	//convert to common type
sl@0
   287
					if (tempElementOrientation == tempCRPOrientation)
sl@0
   288
						{
sl@0
   289
						orientationOK = ETrue;
sl@0
   290
						}
sl@0
   291
					}
sl@0
   292
				if(orientationOK)
sl@0
   293
				    {
sl@0
   294
				    TBool tempElementFlip = aSurfaceConfiguration.Flip();
sl@0
   295
				    TBool tempCRPFlip = aElement.SourceFlipping();
sl@0
   296
				    if(tempElementFlip == tempCRPFlip)
sl@0
   297
				        {
sl@0
   298
				        match = ETrue;
sl@0
   299
				        }
sl@0
   300
				    }
sl@0
   301
				}
sl@0
   302
			}
sl@0
   303
		}
sl@0
   304
	return match;
sl@0
   305
	}
sl@0
   306
sl@0
   307
/** Sets the surfaceconfiguration to the element.
sl@0
   308
sl@0
   309
@param aElement The element to set.
sl@0
   310
@param aSurfaceConfiguration The configuration to set to the element.
sl@0
   311
@param aLimitedSet Set all values if EFalse, otherwise only set viewport and orientation.
sl@0
   312
@return KErrNone on success, error from compositor if it could not set the surface id.
sl@0
   313
*/
sl@0
   314
TInt CWindowElement::SetElement(MWsElement& aElement, const TSurfaceConfiguration& aSurfaceConfiguration, TBool aLimitedSet)
sl@0
   315
	{	//aLimitedSet - EFalse = set all . ETrue = set viewport and orientation only
sl@0
   316
	if (!aLimitedSet)
sl@0
   317
		{
sl@0
   318
		//set surface id
sl@0
   319
		TSurfaceId newSurfaceId;
sl@0
   320
		aSurfaceConfiguration.GetSurfaceId(newSurfaceId);
sl@0
   321
		TInt error = aElement.ConnectSurface(newSurfaceId);
sl@0
   322
		if (error != KErrNone)
sl@0
   323
			{
sl@0
   324
			return error;
sl@0
   325
			}
sl@0
   326
		
sl@0
   327
		//set extent
sl@0
   328
		TRect newExtent;
sl@0
   329
		aSurfaceConfiguration.GetExtent(newExtent);
sl@0
   330
		aElement.SetDestinationRectangle(newExtent);
sl@0
   331
		}
sl@0
   332
	
sl@0
   333
	//set viewport
sl@0
   334
	TRect newViewport;
sl@0
   335
	aSurfaceConfiguration.GetViewport(newViewport);
sl@0
   336
	if (newViewport.iTl.iX < 0 || newViewport.iTl.iY < 0)
sl@0
   337
		{
sl@0
   338
		return KErrArgument;
sl@0
   339
		}
sl@0
   340
	
sl@0
   341
	if (!newViewport.IsEmpty())
sl@0
   342
		{	//need to check if viewport is valid size
sl@0
   343
		TInt error = aElement.SetSourceRectangle(newViewport);
sl@0
   344
        if (error != KErrNone)
sl@0
   345
            {
sl@0
   346
            return error;
sl@0
   347
            }
sl@0
   348
		}
sl@0
   349
sl@0
   350
	//set orientation
sl@0
   351
	//Note for compatibility with Oghma:
sl@0
   352
	//This method does not properly error check or return KErrArgument,
sl@0
   353
	//as that would propagate a client panic in SetBackgroundSurface which did not previously occur.
sl@0
   354
	aElement.SetSourceRotation(GcToElementRotation(aSurfaceConfiguration.Orientation()));
sl@0
   355
	// Set or clear flip if the element flags are changing
sl@0
   356
	if ( (!aSurfaceConfiguration.Flip()) != (!aElement.SourceFlipping()) ) 
sl@0
   357
	    {
sl@0
   358
        aElement.SetSourceFlipping(!aElement.SourceFlipping());
sl@0
   359
	    }
sl@0
   360
	return KErrNone;
sl@0
   361
	}
sl@0
   362
sl@0
   363
/** Attempts to insert a region cache in to the the windows iCache.
sl@0
   364
If it fails, it will remove all cache beyond its index as it is now out of step.
sl@0
   365
sl@0
   366
@param aIndexToInsert The index of where to insert the cache.
sl@0
   367
@return ETrue on success, else EFalse.
sl@0
   368
*/
sl@0
   369
sl@0
   370
TBool CWindowElement::InsertCache(TInt aIndexToInsert)
sl@0
   371
	{
sl@0
   372
	RRegion newRegion;
sl@0
   373
	TInt cacheInsertError = iCache.Insert(newRegion, aIndexToInsert);
sl@0
   374
	if (cacheInsertError == KErrNone)
sl@0
   375
		{
sl@0
   376
		newRegion.Close(); 
sl@0
   377
		return ETrue;	//inserted cache
sl@0
   378
		}
sl@0
   379
	if (iCache.Count() > aIndexToInsert)
sl@0
   380
		{	//if insert failed and there is more cache above where its trying to insert
sl@0
   381
		iCache.Remove(iCache.Count()-1);
sl@0
   382
		cacheInsertError = iCache.Insert(newRegion, aIndexToInsert);
sl@0
   383
		
sl@0
   384
		if (cacheInsertError == KErrNone)
sl@0
   385
			{
sl@0
   386
			newRegion.Close();
sl@0
   387
			return ETrue;	//inserted cache
sl@0
   388
			}
sl@0
   389
		}
sl@0
   390
	
sl@0
   391
	//cant insert this cache, there may be invalid cache to destroy
sl@0
   392
	TInt count = iCache.Count();
sl@0
   393
	while(count > aIndexToInsert)
sl@0
   394
		{	//if there is cache where we wished to insert, it and everything above is now invalid
sl@0
   395
		iCache[count-1].Close();
sl@0
   396
		iCache.Remove(count-1);
sl@0
   397
		count--;
sl@0
   398
		};
sl@0
   399
	newRegion.Close();
sl@0
   400
	return EFalse;	//failed to insert
sl@0
   401
	}
sl@0
   402
sl@0
   403
/** Will send a notification that a surface being unreferenced if it is not
sl@0
   404
found in any WindowElementSet
sl@0
   405
sl@0
   406
@pre Make sure the element being removed has already had its surfaceid set to NULL.
sl@0
   407
@param aSurfaceId The surface being removed
sl@0
   408
*/
sl@0
   409
void CWindowElement::NotifyReleasingSurface(TSurfaceId aSurfaceId)
sl@0
   410
	{
sl@0
   411
	if (CWsTop::SearchDuplicateSurfaceId(aSurfaceId) == EFalse)
sl@0
   412
		{	//NOT found other instance of surface id, send notification now
sl@0
   413
		TWservCrEvent crEvent(TWservCrEvent::ESurfaceUnreferenced, sizeof(aSurfaceId), &aSurfaceId);
sl@0
   414
		TWindowServerEvent::NotifyDrawer(crEvent);
sl@0
   415
		}
sl@0
   416
	}
sl@0
   417
sl@0
   418
/** Called to place a element within the window.
sl@0
   419
It will either create a new element, recycle a element, or extend a element .
sl@0
   420
sl@0
   421
@pre UnassignPlacedElements has already been called.
sl@0
   422
@param aPlacedAttributes Returns the attributes of the surface placed.
sl@0
   423
@param aSurfaceConfiguration The surface configuration for the surface to place.
sl@0
   424
@param aUserDefinedRegion The user defined clipping of the window.
sl@0
   425
@param aScene Access to the scene.
sl@0
   426
@return Serious error, otherwise flags to describe what extra work needs doing.
sl@0
   427
sl@0
   428
@see CWindowElementSet::AssignPlacedElement
sl@0
   429
*/
sl@0
   430
TInt CWindowElement::AssignPlacedElement(
sl@0
   431
		TPlacedAttributes*& aPlacedAttributes,
sl@0
   432
		const TSurfaceConfiguration& aSurfaceConfiguration,
sl@0
   433
		const TRegion& aUserDefinedRegion,
sl@0
   434
		MWsScene& aScene,
sl@0
   435
		TInt	aGcDrawingCount
sl@0
   436
		)
sl@0
   437
	{	
sl@0
   438
	if (iStartGcDrawingCount!=aGcDrawingCount)
sl@0
   439
		{
sl@0
   440
		FlagAssignedElementsDrawnOver(aGcDrawingCount);
sl@0
   441
		}
sl@0
   442
sl@0
   443
	if (iNonRedrawVisibleRegion.CheckError())
sl@0
   444
		{
sl@0
   445
		//CANNOT DO ANYTHING, LEAVE
sl@0
   446
		return KErrNotReady;	//error
sl@0
   447
		}
sl@0
   448
sl@0
   449
	TRect surfaceExtent;
sl@0
   450
	aSurfaceConfiguration.GetExtent(surfaceExtent);
sl@0
   451
	if (!aUserDefinedRegion.Intersects(surfaceExtent))
sl@0
   452
		{
sl@0
   453
		//clippedRegion.Close();
sl@0
   454
		return KErrCancel;	//no intersection, so nothing to do
sl@0
   455
		}
sl@0
   456
sl@0
   457
	TInt matchFound = -1;
sl@0
   458
	
sl@0
   459
	//check 'current' (last placed element) to see if it is a match
sl@0
   460
	if (iHighestReusedIndex != -1)
sl@0
   461
		{
sl@0
   462
		if (IsElementAMatch(*(iPlacedElements[iHighestReusedIndex].iElement),aSurfaceConfiguration,iPlacedElements[iHighestReusedIndex].Flags()) )
sl@0
   463
			{
sl@0
   464
			matchFound = iHighestReusedIndex;
sl@0
   465
			}
sl@0
   466
		}
sl@0
   467
sl@0
   468
	if (matchFound == -1)
sl@0
   469
		{
sl@0
   470
		//check current unassigned elements to see if there is a match
sl@0
   471
		for (TInt ii = (iHighestReusedIndex == -1)?0:iHighestReusedIndex+1; ii < iPlacedElements.Count(); ii++)
sl@0
   472
			{
sl@0
   473
			if (IsElementAMatch(*(iPlacedElements[ii].iElement),aSurfaceConfiguration,iPlacedElements[ii].Flags()))
sl@0
   474
				{
sl@0
   475
				matchFound = ii;
sl@0
   476
				break;
sl@0
   477
				}
sl@0
   478
			}
sl@0
   479
		}
sl@0
   480
	TInt returnValue = 0;
sl@0
   481
	if (matchFound == -1)
sl@0
   482
		{	//not found a element to extend
sl@0
   483
		//iResort = ETrue;
sl@0
   484
		matchFound = 0;
sl@0
   485
		while(1)
sl@0
   486
			{
sl@0
   487
			if (iPlacedElements.Count() == 0 || iHighestReusedIndex == -1 || matchFound == iHighestReusedIndex)	//reached last re assigned element, so cannot reuse one
sl@0
   488
				{
sl@0
   489
				matchFound = -1;
sl@0
   490
				break;
sl@0
   491
				}
sl@0
   492
			if (iPlacedElements[matchFound].Unassigned() && !iPlacedElements[matchFound].LockedConfig())	//found a element to reuse
sl@0
   493
				{
sl@0
   494
				break;
sl@0
   495
				}
sl@0
   496
			matchFound++;
sl@0
   497
			};
sl@0
   498
sl@0
   499
		if (matchFound != -1)
sl@0
   500
			{	//found a element to recycle
sl@0
   501
			returnValue = AssignRecycleElement(matchFound,aSurfaceConfiguration,aUserDefinedRegion,aScene);
sl@0
   502
			}
sl@0
   503
		else
sl@0
   504
			{	//need to create a new element
sl@0
   505
			returnValue = AssignCreateElement(aSurfaceConfiguration,aUserDefinedRegion,aScene);
sl@0
   506
			}
sl@0
   507
		}
sl@0
   508
	else
sl@0
   509
		{	//found a candidate to extend
sl@0
   510
		returnValue = AssignExtendElement(matchFound,aSurfaceConfiguration,aUserDefinedRegion,aScene);
sl@0
   511
		}
sl@0
   512
	if (returnValue < KErrNone)
sl@0
   513
		{	//failed with error
sl@0
   514
			return returnValue;
sl@0
   515
		}
sl@0
   516
	aPlacedAttributes = &iPlacedElements[iHighestReusedIndex];
sl@0
   517
	iFlags|=returnValue;	//update window wide flags
sl@0
   518
	return returnValue;
sl@0
   519
	}
sl@0
   520
sl@0
   521
/** Recycle a element assigned to the window.
sl@0
   522
sl@0
   523
@param aIndex The index of the placed element to recycle.
sl@0
   524
@param aSurfaceConfiguration The surface configuration for the surface to place.
sl@0
   525
@param aUserDefinedRegion The user defined clipping of the window.
sl@0
   526
@param aScene Access to the scene.
sl@0
   527
@return Serious error, otherwise flags to describe what extra work needs doing.
sl@0
   528
sl@0
   529
@see CWindowElement::AssignPlacedElement
sl@0
   530
*/
sl@0
   531
TInt CWindowElement::AssignRecycleElement(const TInt aIndex, const TSurfaceConfiguration& aSurfaceConfiguration, const TRegion& aUserDefinedRegion, MWsScene& aScene)
sl@0
   532
	{
sl@0
   533
	TInt returnValue = 0;
sl@0
   534
	TInt error = KErrNone;
sl@0
   535
	STACK_REGION clippedRegion;
sl@0
   536
	TRect surfaceExtent;
sl@0
   537
	aSurfaceConfiguration.GetExtent(surfaceExtent);
sl@0
   538
	clippedRegion.AddRect(surfaceExtent);
sl@0
   539
	clippedRegion.Intersect(aUserDefinedRegion);
sl@0
   540
	if (clippedRegion.CheckError())
sl@0
   541
		{	//failed crucial calculation, leave while nothing has been changed!
sl@0
   542
		clippedRegion.Close();
sl@0
   543
		return KErrNoMemory;
sl@0
   544
		}
sl@0
   545
	WS_ASSERT_DEBUG(!clippedRegion.IsEmpty(),EWsPanicRegion);
sl@0
   546
	if (clippedRegion.IsEmpty())
sl@0
   547
		{
sl@0
   548
		return KErrArgument;
sl@0
   549
		}
sl@0
   550
	
sl@0
   551
	//keep note of surface being removed
sl@0
   552
	TSurfaceId oldSurfaceId = iPlacedElements[aIndex].iElement->ConnectedSurface();
sl@0
   553
	
sl@0
   554
	//set element with new attributes
sl@0
   555
	error = SetElement(*(iPlacedElements[aIndex].iElement),aSurfaceConfiguration,EFalse);
sl@0
   556
	
sl@0
   557
	if (error < KErrNone)
sl@0
   558
		{	//must be a bad surface id or bad viewport, leave while nothing has been changed!
sl@0
   559
		clippedRegion.Close();
sl@0
   560
		return error;
sl@0
   561
		}
sl@0
   562
	
sl@0
   563
	returnValue|=EFastPath;
sl@0
   564
	returnValue|=EPauseComposition;
sl@0
   565
	//element has been altered, must make sure old surface unreferenced notify will be called
sl@0
   566
	if (!iRemovedSurfacesValid)
sl@0
   567
		{	//didnt reserve space for surface ids, do search immediately
sl@0
   568
		NotifyReleasingSurface(oldSurfaceId);
sl@0
   569
		}
sl@0
   570
	else
sl@0
   571
		{	//keep memory of surface id - search will be done during cleanup. Space already been reserved
sl@0
   572
		iRemovedSurfaces.Append(oldSurfaceId);
sl@0
   573
		}
sl@0
   574
sl@0
   575
	//order placed element correctly in iPlacedElements
sl@0
   576
	TPlacedAttributes tempAttributes = iPlacedElements[aIndex];
sl@0
   577
	iPlacedElements.Remove(aIndex);
sl@0
   578
	//keep cache up to date
sl@0
   579
	if (iCache.Count() > aIndex)
sl@0
   580
		{
sl@0
   581
		iCache[aIndex].Close();
sl@0
   582
		iCache.Remove(aIndex);
sl@0
   583
		}
sl@0
   584
sl@0
   585
	WS_ASSERT_DEBUG(iHighestReusedIndex != -1,EWsPanicRecycleElement);
sl@0
   586
	if (iHighestReusedIndex == -1)
sl@0
   587
		{
sl@0
   588
		return KErrArgument;
sl@0
   589
		}
sl@0
   590
	
sl@0
   591
	error = iPlacedElements.Insert(tempAttributes, iHighestReusedIndex);
sl@0
   592
	if (error < KErrNone)
sl@0
   593
		{	//must destroy the element and leave.  Old elements surface unregistration already dealt with
sl@0
   594
		aScene.DestroySceneElement(tempAttributes.iElement);
sl@0
   595
		clippedRegion.Close();
sl@0
   596
		return KErrNoMemory;
sl@0
   597
		}
sl@0
   598
	returnValue|=EResort;	//element has been moved, will need a resort
sl@0
   599
	//set placed element attributes
sl@0
   600
	iPlacedElements[iHighestReusedIndex].SetUnassigned(EFalse);
sl@0
   601
	iPlacedElements[iHighestReusedIndex].SetLockedConfig(ETrue);
sl@0
   602
	iPlacedElements[iHighestReusedIndex].SetChangedClip(ETrue);
sl@0
   603
	
sl@0
   604
	TRect tempViewport;
sl@0
   605
	aSurfaceConfiguration.GetViewport(tempViewport);
sl@0
   606
sl@0
   607
	TBool cacheSaved = EFalse;
sl@0
   608
	if (iCache.Count() >= iHighestReusedIndex)
sl@0
   609
		{	//need to update cache with visible region of element
sl@0
   610
		TBool insertSuccess = InsertCache(iHighestReusedIndex);
sl@0
   611
		if (insertSuccess != EFalse)
sl@0
   612
			{
sl@0
   613
			iCache[iHighestReusedIndex].Copy(clippedRegion);
sl@0
   614
			if (!iCache[iHighestReusedIndex].CheckError())
sl@0
   615
				{
sl@0
   616
				cacheSaved = ETrue;
sl@0
   617
				}
sl@0
   618
			}
sl@0
   619
		}
sl@0
   620
	if (cacheSaved == EFalse)
sl@0
   621
		{	//need to set clipping to element immediately
sl@0
   622
		TInt count = iCache.Count();
sl@0
   623
		while(count > iHighestReusedIndex)
sl@0
   624
			{	//if there is cache where we wished to insert, it and everything above is now invalid
sl@0
   625
			iCache[count-1].Close();
sl@0
   626
			iCache.Remove(count-1);
sl@0
   627
			count--;
sl@0
   628
			};
sl@0
   629
		}
sl@0
   630
	clippedRegion.Close();
sl@0
   631
	return returnValue;
sl@0
   632
	}
sl@0
   633
sl@0
   634
/** Create a new element for the placed surface.
sl@0
   635
sl@0
   636
@param aSurfaceConfiguration The surface configuration for the surface to place.
sl@0
   637
@param aUserDefinedRegion The user defined clipping of the window.
sl@0
   638
@param aScene Access to the scene.
sl@0
   639
@return Serious error, otherwise flags to describe what extra work needs doing.
sl@0
   640
sl@0
   641
@see CWindowElement::AssignPlacedElement
sl@0
   642
*/
sl@0
   643
TInt CWindowElement::AssignCreateElement(const TSurfaceConfiguration& aSurfaceConfiguration, const TRegion& aUserDefinedRegion, MWsScene& aScene)
sl@0
   644
	{
sl@0
   645
	TInt returnValue = 0;
sl@0
   646
	TInt error = KErrNone;
sl@0
   647
	STACK_REGION clippedRegion;
sl@0
   648
	TRect surfaceExtent;
sl@0
   649
	aSurfaceConfiguration.GetExtent(surfaceExtent);
sl@0
   650
	//calculate region
sl@0
   651
	clippedRegion.AddRect(surfaceExtent);
sl@0
   652
	clippedRegion.Intersect(aUserDefinedRegion);
sl@0
   653
	if (clippedRegion.CheckError())
sl@0
   654
		{	//failed, return KErrNoMemory
sl@0
   655
		clippedRegion.Close();
sl@0
   656
		return KErrNoMemory;
sl@0
   657
		}
sl@0
   658
	
sl@0
   659
	TPlacedAttributes newElement;
sl@0
   660
	if (iHighestReusedIndex == -1)
sl@0
   661
		{
sl@0
   662
		error = iPlacedElements.Insert(newElement,0);
sl@0
   663
		}
sl@0
   664
	else
sl@0
   665
		{	//insert above current highest reused element
sl@0
   666
		error = iPlacedElements.Insert(newElement,iHighestReusedIndex+1);
sl@0
   667
		}
sl@0
   668
	if (error >= KErrNone)
sl@0
   669
		{
sl@0
   670
		iHighestReusedIndex++;
sl@0
   671
		}
sl@0
   672
	else
sl@0
   673
		{
sl@0
   674
		//FAILED! LEAVE FUNCTION. nothing has been changed
sl@0
   675
		clippedRegion.Close();
sl@0
   676
		return KErrNoMemory;
sl@0
   677
		}
sl@0
   678
sl@0
   679
	//create new element with new configuration
sl@0
   680
	MWsElement* element = NULL;
sl@0
   681
	TRAP(error,element = aScene.CreateSceneElementL());
sl@0
   682
	if (error != KErrNone)
sl@0
   683
		{
sl@0
   684
		//FAILED! LEAVE FUNCTION. nothing has been changed
sl@0
   685
		clippedRegion.Close();
sl@0
   686
		return error;
sl@0
   687
		}
sl@0
   688
sl@0
   689
	element->SetGlobalAlpha( 0xFF );
sl@0
   690
sl@0
   691
	iPlacedElements[iHighestReusedIndex].iElement = element;
sl@0
   692
	error = SetElement(*element,aSurfaceConfiguration,EFalse);
sl@0
   693
	if (error < KErrNone)
sl@0
   694
		{
sl@0
   695
		//must be a bad surface id or viewport. Leave now.
sl@0
   696
		aScene.DestroySceneElement(element);
sl@0
   697
		clippedRegion.Close();
sl@0
   698
		iPlacedElements.Remove(iHighestReusedIndex);
sl@0
   699
		iHighestReusedIndex--;
sl@0
   700
		return error;
sl@0
   701
		}
sl@0
   702
	
sl@0
   703
	//Now certain that these must be flagged
sl@0
   704
	returnValue|=EFastPath;
sl@0
   705
	returnValue|=EPauseComposition;
sl@0
   706
	returnValue|=EResort;
sl@0
   707
sl@0
   708
	//set placed element attributes
sl@0
   709
	iPlacedElements[iHighestReusedIndex].SetUnassigned(EFalse);
sl@0
   710
	iPlacedElements[iHighestReusedIndex].SetChangedClip(ETrue);
sl@0
   711
	iPlacedElements[iHighestReusedIndex].SetLockedConfig(ETrue);
sl@0
   712
	
sl@0
   713
	TRect tempViewport;
sl@0
   714
	aSurfaceConfiguration.GetViewport(tempViewport);
sl@0
   715
sl@0
   716
	TBool cacheSaved = EFalse;
sl@0
   717
	if (iCache.Count() >= iHighestReusedIndex)
sl@0
   718
		{	//should try to add new elements region to cache
sl@0
   719
		TBool insertSuccess = InsertCache(iHighestReusedIndex);
sl@0
   720
		if (insertSuccess != EFalse)
sl@0
   721
			{
sl@0
   722
			iCache[iHighestReusedIndex].Copy(clippedRegion);
sl@0
   723
			if (!iCache[iHighestReusedIndex].CheckError())
sl@0
   724
				{
sl@0
   725
				cacheSaved = ETrue;
sl@0
   726
				}
sl@0
   727
			}
sl@0
   728
		}
sl@0
   729
	if (cacheSaved == EFalse)
sl@0
   730
		{	//need to clear cache from this point onward, and set clipping to element immediately
sl@0
   731
		TInt count = iCache.Count();
sl@0
   732
		while(count > iHighestReusedIndex)
sl@0
   733
			{	//if there is cache where we wished to insert, it and everything above is now invalid
sl@0
   734
			iCache[count-1].Close();
sl@0
   735
			iCache.Remove(count-1);
sl@0
   736
			count--;
sl@0
   737
			};
sl@0
   738
		}
sl@0
   739
	//add the new element to the scene
sl@0
   740
    error = aScene.InsertSceneElement(element, NULL);    //place at the back. will get ordered correctly later
sl@0
   741
	WS_ASSERT_DEBUG(error == KErrNone,EWsPanicSceneErrorIgnored);
sl@0
   742
	clippedRegion.Close();
sl@0
   743
	return returnValue;
sl@0
   744
	}
sl@0
   745
sl@0
   746
/** Extend an existing placed surface element.
sl@0
   747
sl@0
   748
@param The index of the element to extend.
sl@0
   749
@param aSurfaceConfiguration The surface configuration for the surface to place.
sl@0
   750
@param aUserDefinedRegion The user defined clipping of the window.
sl@0
   751
@param aScene Access to the scene.
sl@0
   752
@return Serious error, otherwise flags to describe what extra work needs doing.
sl@0
   753
sl@0
   754
@see CWindowElement::AssignPlacedElement
sl@0
   755
*/
sl@0
   756
TInt CWindowElement::AssignExtendElement(const TInt aIndex, const TSurfaceConfiguration& aSurfaceConfiguration, const TRegion& aUserDefinedRegion, MWsScene& /*aScene*/)
sl@0
   757
	{
sl@0
   758
	TInt returnValue = 0;
sl@0
   759
	TInt error = KErrNone;
sl@0
   760
	STACK_REGION backupRegionAgainstFail;
sl@0
   761
	STACK_REGION clippedRegion;
sl@0
   762
	TRect surfaceExtent;
sl@0
   763
	aSurfaceConfiguration.GetExtent(surfaceExtent);
sl@0
   764
	TRect surfaceViewport;
sl@0
   765
	aSurfaceConfiguration.GetViewport(surfaceViewport);
sl@0
   766
	
sl@0
   767
	if (!iPlacedElements[aIndex].LockedConfig())
sl@0
   768
		{	//set element now. If viewport is invalid its ok to fail now
sl@0
   769
		error = SetElement(*iPlacedElements[aIndex].iElement,aSurfaceConfiguration,ETrue);
sl@0
   770
		if (error < KErrNone)
sl@0
   771
			{
sl@0
   772
			backupRegionAgainstFail.Close();
sl@0
   773
			clippedRegion.Close();
sl@0
   774
			return error;
sl@0
   775
			}
sl@0
   776
		}
sl@0
   777
sl@0
   778
	if(iPlacedElements[aIndex].Unassigned() && !iPlacedElements[aIndex].Overlapping())
sl@0
   779
		{
sl@0
   780
		//backup stays clear
sl@0
   781
		clippedRegion.Copy(aUserDefinedRegion);
sl@0
   782
		}
sl@0
   783
	else
sl@0
   784
		{
sl@0
   785
		clippedRegion.Copy(iNonRedrawVisibleRegion);
sl@0
   786
		if (iCache.Count() > aIndex)	//if cache is ok
sl@0
   787
			{
sl@0
   788
			clippedRegion.Intersect(iCache[aIndex]);
sl@0
   789
			clippedRegion.Union(iCache[aIndex]);	//previously added regions should be kept
sl@0
   790
			}
sl@0
   791
		else
sl@0
   792
			{
sl@0
   793
			STACK_REGION tempSurfaceBaseArea;
sl@0
   794
            TRect destinationRect;
sl@0
   795
            iPlacedElements[aIndex].iElement->GetDestinationRectangle(destinationRect);
sl@0
   796
            tempSurfaceBaseArea.AddRect(destinationRect);
sl@0
   797
			tempSurfaceBaseArea.Offset(surfaceExtent.iTl);
sl@0
   798
			clippedRegion.Intersect(tempSurfaceBaseArea);
sl@0
   799
			clippedRegion.Union(tempSurfaceBaseArea);
sl@0
   800
			tempSurfaceBaseArea.Close();
sl@0
   801
			}
sl@0
   802
		backupRegionAgainstFail.Copy(clippedRegion);
sl@0
   803
		//clipped region should contain something
sl@0
   804
		clippedRegion.Union(aUserDefinedRegion);
sl@0
   805
		}
sl@0
   806
	clippedRegion.ClipRect(surfaceExtent);
sl@0
   807
	
sl@0
   808
	if (backupRegionAgainstFail.CheckError() || clippedRegion.CheckError())
sl@0
   809
		{	//failed critical calculations, leave now before anything has been changed
sl@0
   810
		clippedRegion.Close();
sl@0
   811
		backupRegionAgainstFail.Close();
sl@0
   812
		return KErrNoMemory;
sl@0
   813
		}
sl@0
   814
sl@0
   815
	TBool setCache = EFalse;
sl@0
   816
	if (iCache.Count() > aIndex)	//if Cache is ok
sl@0
   817
		{
sl@0
   818
		//compare
sl@0
   819
		TInt compareReturn;
sl@0
   820
		compareReturn = TRegionExtend::Cast(iCache[aIndex]).TestDifference(clippedRegion);
sl@0
   821
sl@0
   822
		if (compareReturn&TRegionExtend::EDiffers)	//element has changed
sl@0
   823
			{
sl@0
   824
			iPlacedElements[aIndex].SetChangedClip(ETrue);
sl@0
   825
			if (compareReturn&TRegionExtend::EAdd)	//element has become revealed on the screen
sl@0
   826
				{
sl@0
   827
				returnValue|=EPauseComposition;	//need to pause before the end of assign placed element
sl@0
   828
				}
sl@0
   829
			else
sl@0
   830
				{
sl@0
   831
				returnValue|=EFastPath;	//fastpath and pause will be called during cleanup
sl@0
   832
				}
sl@0
   833
			}
sl@0
   834
		
sl@0
   835
		//copy clipped region to cache
sl@0
   836
		iCache[aIndex].Copy(clippedRegion);	//can fail
sl@0
   837
		setCache = ETrue;
sl@0
   838
		if (iCache[aIndex].CheckError())
sl@0
   839
			{	//copy failed, remove cache from this element onwards
sl@0
   840
			TInt count = iCache.Count();
sl@0
   841
			while(count > aIndex)
sl@0
   842
				{	//if there is cache where we wished to insert, it and everything above is now invalid
sl@0
   843
				iCache[count-1].Close();
sl@0
   844
				iCache.Remove(count-1);
sl@0
   845
				count--;
sl@0
   846
				};
sl@0
   847
			setCache = EFalse;
sl@0
   848
			}
sl@0
   849
		}
sl@0
   850
	if (setCache == EFalse)
sl@0
   851
		{	//need to pause composition and update element immediately
sl@0
   852
		returnValue|=EPauseComposition;
sl@0
   853
		}
sl@0
   854
	iPlacedElements[aIndex].SetUnassigned(EFalse);
sl@0
   855
	iPlacedElements[aIndex].SetLockedConfig(ETrue);
sl@0
   856
	iHighestReusedIndex = aIndex;
sl@0
   857
	backupRegionAgainstFail.Close();
sl@0
   858
	clippedRegion.Close();
sl@0
   859
	return returnValue;
sl@0
   860
	}
sl@0
   861
sl@0
   862
/** Called at the end of a redraw to set cached changed to elements, and
sl@0
   863
removed elements that are still unassigned.
sl@0
   864
sl@0
   865
@param aScene Access to the scene.
sl@0
   866
@return Flags to describe what extra work needs doing.
sl@0
   867
@see CWindowElementSet::CleanUpPlacedElements
sl@0
   868
*/
sl@0
   869
TInt CWindowElement::CleanUpPlacedElements(MWsScene& aScene, TInt aGcDrawingCount)
sl@0
   870
	{
sl@0
   871
	if (iStartGcDrawingCount!=aGcDrawingCount)
sl@0
   872
		{
sl@0
   873
		FlagAssignedElementsDrawnOver(aGcDrawingCount);
sl@0
   874
		}
sl@0
   875
	TInt returnFlags = iFlags;
sl@0
   876
	if (iFlags > 0)
sl@0
   877
		{	//copy flags to return, and ensure pause is flagged
sl@0
   878
		returnFlags|=EPauseComposition;
sl@0
   879
		}
sl@0
   880
	TInt ii;
sl@0
   881
	TBool removeElements = EFalse;
sl@0
   882
	//destroy placed element reference
sl@0
   883
	for (ii=0; ii<iPlacedElements.Count(); ii++)
sl@0
   884
		{
sl@0
   885
		if (iPlacedElements[ii].Unassigned() && !iPlacedElements[ii].Overlapping())
sl@0
   886
			{
sl@0
   887
			returnFlags|=EFastPath;	//need to invoke 1108
sl@0
   888
			if (iRemovedSurfacesValid)
sl@0
   889
				{
sl@0
   890
				iRemovedSurfaces.Append(iPlacedElements[ii].iElement->ConnectedSurface());
sl@0
   891
				iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
sl@0
   892
				}
sl@0
   893
			else
sl@0
   894
				{
sl@0
   895
				TSurfaceId tempId = iPlacedElements[ii].iElement->ConnectedSurface();
sl@0
   896
				iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
sl@0
   897
				NotifyReleasingSurface(tempId);
sl@0
   898
				}
sl@0
   899
			removeElements = ETrue;
sl@0
   900
			}
sl@0
   901
		}
sl@0
   902
sl@0
   903
	if (removeElements)
sl@0
   904
		{	//there will be elements to remove
sl@0
   905
		for (ii=0; ii<iPlacedElements.Count(); ii++)
sl@0
   906
			{
sl@0
   907
			if (iPlacedElements[ii].iElement->ConnectedSurface().IsNull())
sl@0
   908
				{
sl@0
   909
				aScene.DestroySceneElement(iPlacedElements[ii].iElement);
sl@0
   910
				iPlacedElements.Remove(ii);
sl@0
   911
				if (iCache.Count() > ii)
sl@0
   912
					{	//there is cache for this element, remove it
sl@0
   913
					iCache[ii].Close();
sl@0
   914
					iCache.Remove(ii);
sl@0
   915
					}
sl@0
   916
				ii--;
sl@0
   917
				}
sl@0
   918
			}
sl@0
   919
		}
sl@0
   920
sl@0
   921
	if (iRemovedSurfaces.Count() > 1)
sl@0
   922
		{	//remove duplicates of surface id from removal list
sl@0
   923
		for (ii = 0; ii < iRemovedSurfaces.Count() - 1; ii++)
sl@0
   924
			{
sl@0
   925
			for (TInt jj = ii+1; jj < iRemovedSurfaces.Count(); jj++)
sl@0
   926
				{
sl@0
   927
				if (iRemovedSurfaces[ii] == iRemovedSurfaces[jj])
sl@0
   928
					{
sl@0
   929
					iRemovedSurfaces.Remove(jj);
sl@0
   930
					jj--;
sl@0
   931
					}
sl@0
   932
				}
sl@0
   933
			}
sl@0
   934
		}
sl@0
   935
	
sl@0
   936
	//do global search for surface ids
sl@0
   937
	while (iRemovedSurfaces.Count()>0)
sl@0
   938
		{
sl@0
   939
		TInt tempRemoval = iRemovedSurfaces.Count() - 1;
sl@0
   940
		NotifyReleasingSurface(iRemovedSurfaces[tempRemoval]);
sl@0
   941
		iRemovedSurfaces.Remove(tempRemoval);
sl@0
   942
		};
sl@0
   943
sl@0
   944
	iRemovedSurfaces.Reset();
sl@0
   945
sl@0
   946
	//clip unassigned overlapping entries, mark as changedclip if region changed
sl@0
   947
	for (ii=0; ii<iPlacedElements.Count(); ii++)
sl@0
   948
		{
sl@0
   949
		if (iPlacedElements[ii].Unassigned())	//non overlapping unassigned have already been destroyed
sl@0
   950
			{	//these must overlap
sl@0
   951
			TBool failureOccured = EFalse;
sl@0
   952
			if (iCache.Count() > ii)
sl@0
   953
				{
sl@0
   954
				//if cache region is entirely inside non redraw vis region, nothing to update!
sl@0
   955
				TInt changed = 
sl@0
   956
						TRegionExtend::Cast(iCache[ii]).TestDifference(iNonRedrawVisibleRegion);
sl@0
   957
				if (changed&TRegionExtend::ESub)
sl@0
   958
					{	//the clipping will change
sl@0
   959
					iCache[ii].Intersect(iNonRedrawVisibleRegion);
sl@0
   960
					if (iCache[ii].CheckError())
sl@0
   961
						{
sl@0
   962
						failureOccured = ETrue;
sl@0
   963
						iCache[ii].Close();
sl@0
   964
						iCache.Remove(ii);
sl@0
   965
						}
sl@0
   966
					else
sl@0
   967
						{
sl@0
   968
						iPlacedElements[ii].SetChangedClip(ETrue);	//changed clipping
sl@0
   969
						}
sl@0
   970
					}
sl@0
   971
				}
sl@0
   972
			else
sl@0
   973
				{	//attempt getting element region to perform calculation and update element directly
sl@0
   974
				STACK_REGION elementRegion;
sl@0
   975
	            TRect destinationRect;
sl@0
   976
	            iPlacedElements[ii].iElement->GetDestinationRectangle(destinationRect);
sl@0
   977
	            elementRegion.AddRect(destinationRect);
sl@0
   978
				if (elementRegion.CheckError())
sl@0
   979
					{
sl@0
   980
					failureOccured = ETrue;
sl@0
   981
					}
sl@0
   982
				else
sl@0
   983
					{
sl@0
   984
					TRect elementExtent;
sl@0
   985
					iPlacedElements[ii].iElement->GetDestinationRectangle(elementExtent);
sl@0
   986
					elementRegion.Offset(elementExtent.iTl);	//get element region into absolute coordinates
sl@0
   987
					TInt changed =
sl@0
   988
						TRegionExtend::Cast(elementRegion).TestDifference(iNonRedrawVisibleRegion);
sl@0
   989
					if (changed&TRegionExtend::ESub)
sl@0
   990
						{	//need to clip element back
sl@0
   991
						elementRegion.Intersect(iNonRedrawVisibleRegion);
sl@0
   992
						if (!elementRegion.CheckError())
sl@0
   993
							{
sl@0
   994
							elementRegion.Offset(-elementExtent.iTl);	//put element region back into relative coords
sl@0
   995
							returnFlags|=EFastPath;
sl@0
   996
							}
sl@0
   997
						else
sl@0
   998
							{
sl@0
   999
							failureOccured = ETrue;
sl@0
  1000
							}
sl@0
  1001
						}
sl@0
  1002
					}
sl@0
  1003
				elementRegion.Close();
sl@0
  1004
				}
sl@0
  1005
			if (failureOccured)
sl@0
  1006
				{	//need to release element
sl@0
  1007
				TSurfaceId tempId = iPlacedElements[ii].iElement->ConnectedSurface();
sl@0
  1008
				iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
sl@0
  1009
				NotifyReleasingSurface(tempId);
sl@0
  1010
				aScene.DestroySceneElement(iPlacedElements[ii].iElement);
sl@0
  1011
				returnFlags|=EFailed;
sl@0
  1012
				iPlacedElements.Remove(ii);	//remove placed element entry, cache is already removed
sl@0
  1013
				ii--;
sl@0
  1014
				}
sl@0
  1015
			}
sl@0
  1016
		}
sl@0
  1017
sl@0
  1018
	//update any elements marked as changedclip , clear all cache, clear flags
sl@0
  1019
	for (ii=0; ii<iPlacedElements.Count(); ii++)
sl@0
  1020
		{
sl@0
  1021
		if (iPlacedElements[ii].ChangedClip())
sl@0
  1022
			{
sl@0
  1023
			returnFlags|=EFastPath;	//need to invoke 1108
sl@0
  1024
			}
sl@0
  1025
		else
sl@0
  1026
			if (	!(returnFlags&EFastPath)
sl@0
  1027
				&&	iPlacedElements[ii].LockedConfig() && !iPlacedElements[ii].Unassigned()
sl@0
  1028
				&&	iPlacedElements[ii].DrawnOver()!=iPlacedElements[ii].DrawnOverLast()	)
sl@0
  1029
						{
sl@0
  1030
						returnFlags|=EFastPath;
sl@0
  1031
						}
sl@0
  1032
		iPlacedElements[ii].ClearRedrawFlags();
sl@0
  1033
		if (iCache.Count() > ii)
sl@0
  1034
			{
sl@0
  1035
			iCache[ii].Clear();
sl@0
  1036
			}
sl@0
  1037
		}
sl@0
  1038
	iCache.Reset();
sl@0
  1039
	iNonRedrawVisibleRegion.Clear();
sl@0
  1040
	iHighestReusedIndex = -1;
sl@0
  1041
	return returnFlags;
sl@0
  1042
	}
sl@0
  1043
sl@0
  1044
/** Move all elements in this windowelement to above the supplied element in the scene
sl@0
  1045
sl@0
  1046
@param aScene Access to the scene.
sl@0
  1047
@param aElement The element to set all this windows elements above.
sl@0
  1048
*/
sl@0
  1049
sl@0
  1050
void CWindowElement::MoveToAboveGivenElement(MWsScene& aScene, MWsElement* aElement)
sl@0
  1051
	{
sl@0
  1052
	MWsElement* newAboveElement = iBackgroundElement.iElement;
sl@0
  1053
	MWsElement* previousAboveElement = aElement;
sl@0
  1054
	if (newAboveElement)
sl@0
  1055
		{
sl@0
  1056
		aScene.InsertSceneElement(newAboveElement, previousAboveElement);	//background element is put on first
sl@0
  1057
		previousAboveElement = newAboveElement;		
sl@0
  1058
		}
sl@0
  1059
	TInt placedElements = iPlacedElements.Count();
sl@0
  1060
	if (placedElements != 0)
sl@0
  1061
		{
sl@0
  1062
		for (TInt ii = 0; ii < placedElements; ii++)
sl@0
  1063
			{
sl@0
  1064
			newAboveElement = iPlacedElements[ii].iElement;
sl@0
  1065
			aScene.InsertSceneElement(newAboveElement, previousAboveElement);	//place element above previous above element
sl@0
  1066
			previousAboveElement = newAboveElement;	
sl@0
  1067
			}
sl@0
  1068
		}
sl@0
  1069
	}
sl@0
  1070
sl@0
  1071
/** Updates the elements extent, whether from a window movement or a change in window size
sl@0
  1072
sl@0
  1073
@param aOffset The movement of the window.  If NULL then the window has changed size.
sl@0
  1074
@see CWindowElementSet::UpdateElementExtent
sl@0
  1075
*/
sl@0
  1076
void CWindowElement::UpdateElementExtent(const TPoint* aOffset)
sl@0
  1077
	{
sl@0
  1078
	if (aOffset)	//window moved
sl@0
  1079
		{
sl@0
  1080
		TRect tempExtent;
sl@0
  1081
		MWsElement* element = iBackgroundElement.iElement;
sl@0
  1082
		TBool complete = EFalse;
sl@0
  1083
		TInt placedElementDone = -1;
sl@0
  1084
		while (!complete)
sl@0
  1085
			{
sl@0
  1086
			if (!element && placedElementDone < iPlacedElements.Count()-1)
sl@0
  1087
				{
sl@0
  1088
				placedElementDone++;
sl@0
  1089
				element = iPlacedElements[placedElementDone].iElement;
sl@0
  1090
				}
sl@0
  1091
			if (!element)
sl@0
  1092
				{
sl@0
  1093
				complete = ETrue;
sl@0
  1094
				}
sl@0
  1095
			else
sl@0
  1096
				{
sl@0
  1097
				tempExtent = const_cast<CWsClientWindow&>(iWindow).GetOriginalDestElementRect();
sl@0
  1098
				tempExtent.Move(*aOffset);
sl@0
  1099
				element->SetDestinationRectangle(tempExtent);
sl@0
  1100
				const_cast<CWsClientWindow&>(iWindow).SetOriginalDestElementRect(tempExtent);
sl@0
  1101
				}
sl@0
  1102
			element = NULL;
sl@0
  1103
			}
sl@0
  1104
		}
sl@0
  1105
	else			//window changed size
sl@0
  1106
		{
sl@0
  1107
		if (!iBackgroundElement.ExplicitExtent() && iBackgroundElement.iElement)
sl@0
  1108
			{
sl@0
  1109
			iBackgroundElement.iElement->SetDestinationRectangle(iWindow.FullRect());
sl@0
  1110
			const_cast<CWsClientWindow&>(iWindow).SetOriginalDestElementRect(iWindow.FullRect());
sl@0
  1111
			}
sl@0
  1112
		}
sl@0
  1113
	}
sl@0
  1114
sl@0
  1115
/** Checks the windows placed elements when the windows visibility is changed.
sl@0
  1116
If the placed elements are no longer visible, they are removed.
sl@0
  1117
sl@0
  1118
@param aRegion The new visible region of the window.
sl@0
  1119
@param aScene Access to the scene.
sl@0
  1120
@return ETrue if any elements have been removed, otherwise EFalse.
sl@0
  1121
@see CWindowElementSet::SetVisibleRegion
sl@0
  1122
*/
sl@0
  1123
TBool CWindowElement::SetVisibleRegion(const TRegion& aRegion, MWsScene& aScene)
sl@0
  1124
	{
sl@0
  1125
	TBool retcode=EFalse;
sl@0
  1126
	if (iPlacedElements.Count() == 0)
sl@0
  1127
		{
sl@0
  1128
		return EFalse;	//there is noting to do
sl@0
  1129
		}
sl@0
  1130
	
sl@0
  1131
	STACK_REGION tempRegion;
sl@0
  1132
	for (TInt ii = 0; ii < iPlacedElements.Count(); ii++)
sl@0
  1133
		{
sl@0
  1134
		TRect tempExtent;
sl@0
  1135
        tempRegion.Clear();
sl@0
  1136
        iPlacedElements[ii].iElement->GetDestinationRectangle(tempExtent);
sl@0
  1137
        tempRegion.AddRect(tempExtent);
sl@0
  1138
		if (tempRegion.CheckError())	//if there was error getting region
sl@0
  1139
			{
sl@0
  1140
			tempRegion.Clear();
sl@0
  1141
			tempRegion.AddRect(tempExtent);
sl@0
  1142
			}
sl@0
  1143
		else
sl@0
  1144
			{	//offset basearea of element
sl@0
  1145
			tempRegion.Offset(tempExtent.iTl);
sl@0
  1146
			}
sl@0
  1147
		TInt regionReturn = TRegionExtend::Cast(aRegion).TestDifference(tempRegion);
sl@0
  1148
		if (regionReturn&TRegionExtend::ENoIntersect)
sl@0
  1149
			{	//placed surface needs to be removed
sl@0
  1150
			TSurfaceId tempId = iPlacedElements[ii].iElement->ConnectedSurface();
sl@0
  1151
			iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
sl@0
  1152
			NotifyReleasingSurface(tempId);
sl@0
  1153
			aScene.DestroySceneElement(iPlacedElements[ii].iElement);
sl@0
  1154
			retcode = ETrue;	//a element has been removed
sl@0
  1155
			iPlacedElements.Remove(ii);
sl@0
  1156
			if (iCache.Count() > ii)
sl@0
  1157
				{	//keep cache up to date
sl@0
  1158
				iCache[ii].Close();
sl@0
  1159
				iCache.Remove(ii);
sl@0
  1160
				}
sl@0
  1161
			ii--;
sl@0
  1162
			}
sl@0
  1163
		}
sl@0
  1164
	tempRegion.Close();
sl@0
  1165
	return retcode;
sl@0
  1166
	}
sl@0
  1167
sl@0
  1168
/** Checks if any of the windows element ids match the one in question.
sl@0
  1169
sl@0
  1170
@param aSurfaceId The surface id to match against.
sl@0
  1171
@return ETrue if a match is found, otherwise EFalse.
sl@0
  1172
sl@0
  1173
@see CWindowElementSet::SearchDuplicateSurfaceId
sl@0
  1174
*/
sl@0
  1175
TBool CWindowElement::SearchDuplicateSurfaceId(const TSurfaceId& aSurfaceId)
sl@0
  1176
	{
sl@0
  1177
	if (iBackgroundElement.iElement)
sl@0
  1178
		{
sl@0
  1179
		if (iBackgroundElement.iElement->ConnectedSurface() == aSurfaceId)
sl@0
  1180
			{
sl@0
  1181
			return ETrue;
sl@0
  1182
			}
sl@0
  1183
		}
sl@0
  1184
	if (iPlacedElements.Count() > 0)
sl@0
  1185
		{
sl@0
  1186
		for (TInt ii = 0; ii < iPlacedElements.Count(); ii++)
sl@0
  1187
			{
sl@0
  1188
			if (iPlacedElements[ii].iElement)	//if removed without reserved space for ids, could be a null element
sl@0
  1189
				{
sl@0
  1190
				if (iPlacedElements[ii].iElement->ConnectedSurface() == aSurfaceId)
sl@0
  1191
					{
sl@0
  1192
					return ETrue;
sl@0
  1193
					}
sl@0
  1194
				}
sl@0
  1195
			}
sl@0
  1196
		}
sl@0
  1197
	return EFalse;
sl@0
  1198
	}
sl@0
  1199
sl@0
  1200
/** Sets the opacity for the background surface and if setting to 0 removes placed surfaces.
sl@0
  1201
sl@0
  1202
@param aOpacity The opacity to set.
sl@0
  1203
@param aScene Access to the scene.
sl@0
  1204
@see CWindowElementSet::SetElementOpacity
sl@0
  1205
*/
sl@0
  1206
void CWindowElement::SetElementOpacity(TInt aOpacity, MWsScene& aScene)
sl@0
  1207
	{
sl@0
  1208
	if (iBackgroundElement.iElement)
sl@0
  1209
		{
sl@0
  1210
	    TUint32 flags = 0;
sl@0
  1211
	    iBackgroundElement.iElement->GetTargetRendererFlags(flags);
sl@0
  1212
	    flags |= MWsElement::EElementTransparencyGlobalAlpha;
sl@0
  1213
		iBackgroundElement.iElement->SetTargetRendererFlags(MWsElement::EElementTransparencyGlobalAlpha);
sl@0
  1214
		iBackgroundElement.iElement->SetGlobalAlpha(aOpacity);
sl@0
  1215
		iBackgroundElement.SetConcealed(aOpacity==0);
sl@0
  1216
		}
sl@0
  1217
	if (aOpacity == 0)
sl@0
  1218
		{
sl@0
  1219
		while (iPlacedElements.Count() > 0)
sl@0
  1220
			{	//remove any placed elements
sl@0
  1221
			TInt placedElement = iPlacedElements.Count()-1;
sl@0
  1222
			TSurfaceId tempId = iPlacedElements[placedElement].iElement->ConnectedSurface();
sl@0
  1223
			iPlacedElements[placedElement].iElement->ConnectSurface(TSurfaceId::CreateNullId());
sl@0
  1224
			NotifyReleasingSurface(tempId);
sl@0
  1225
			aScene.DestroySceneElement(iPlacedElements[placedElement].iElement);
sl@0
  1226
			iPlacedElements.Remove(placedElement);
sl@0
  1227
			};
sl@0
  1228
		}
sl@0
  1229
	}
sl@0
  1230
sl@0
  1231
MWsElement* CWindowElement::Element() const
sl@0
  1232
    {
sl@0
  1233
    return iBackgroundElement.iElement;
sl@0
  1234
    }
sl@0
  1235
sl@0
  1236
//
sl@0
  1237
// CWindowElementSet
sl@0
  1238
//
sl@0
  1239
sl@0
  1240
/** Destroys the set of window-element pairs.
sl@0
  1241
*/
sl@0
  1242
CWindowElementSet::~CWindowElementSet()
sl@0
  1243
	{
sl@0
  1244
	ASSERT(iElements.Count() == 0);
sl@0
  1245
	iElements.Close();
sl@0
  1246
	}
sl@0
  1247
sl@0
  1248
/** Basic NewL constructor
sl@0
  1249
@param aScene To allow access to the scene.
sl@0
  1250
@param aComposer To allow access to the composer.
sl@0
  1251
*/
sl@0
  1252
CWindowElementSet* CWindowElementSet::NewL(MWsScene& aScene)
sl@0
  1253
	{
sl@0
  1254
	CWindowElementSet* wls = new (ELeave) CWindowElementSet(aScene);
sl@0
  1255
	return wls;
sl@0
  1256
	}
sl@0
  1257
sl@0
  1258
/** Sets a new background surface on the window supplied.
sl@0
  1259
Will remove any previous background surface.
sl@0
  1260
sl@0
  1261
@param aWindow The window to place a background surface on.
sl@0
  1262
@return A reference to the new background surface attributes.
sl@0
  1263
*/
sl@0
  1264
TBackgroundAttributes& CWindowElementSet::AcquireBackgroundElementL(CWsClientWindow& aWindow)
sl@0
  1265
	{
sl@0
  1266
	// Find the matching element.
sl@0
  1267
	TInt index;
sl@0
  1268
	TInt result = FindEntry(aWindow, index);
sl@0
  1269
	MWsElement* element;
sl@0
  1270
sl@0
  1271
	// If a background element is already associated with the window, then unregister
sl@0
  1272
	// the surface. Create and add a new element to scene.
sl@0
  1273
	// This will ensure that the element has default values.
sl@0
  1274
	if (result != KErrNotFound && iElements[index]->iBackgroundElement.iElement)
sl@0
  1275
	    {
sl@0
  1276
	    element = iElements[index]->iBackgroundElement.iElement;
sl@0
  1277
		TSurfaceId surface = element->ConnectedSurface();
sl@0
  1278
		element->ConnectSurface(TSurfaceId::CreateNullId());
sl@0
  1279
		UnregisterSurface(surface);
sl@0
  1280
		iScene.DestroySceneElement(element);
sl@0
  1281
	    }
sl@0
  1282
	if (index < 0)
sl@0
  1283
		{
sl@0
  1284
		index = 0;
sl@0
  1285
		}
sl@0
  1286
sl@0
  1287
	// Allocate a new element and add it to the set and the scene
sl@0
  1288
	element = iScene.CreateSceneElementL();
sl@0
  1289
	
sl@0
  1290
	TInt returnCode;
sl@0
  1291
	if (result == KErrNotFound)
sl@0
  1292
		{
sl@0
  1293
		CWindowElement* winelement = new CWindowElement(aWindow);
sl@0
  1294
		if (!winelement)
sl@0
  1295
			{
sl@0
  1296
			iScene.DestroySceneElement(element);
sl@0
  1297
			User::Leave(KErrNoMemory);
sl@0
  1298
			}
sl@0
  1299
		returnCode = iElements.Insert(winelement, index);
sl@0
  1300
		if(returnCode != KErrNone)
sl@0
  1301
		    {
sl@0
  1302
		    delete winelement;
sl@0
  1303
		    iScene.DestroySceneElement(element);
sl@0
  1304
		    User::Leave(returnCode);
sl@0
  1305
		    }
sl@0
  1306
		}
sl@0
  1307
sl@0
  1308
	// Use the element below the insertion point to decide where the element
sl@0
  1309
	// goes in the scene
sl@0
  1310
	returnCode = KErrNone;
sl@0
  1311
	if (index == 0)
sl@0
  1312
		{
sl@0
  1313
		returnCode = iScene.InsertSceneElement(element, NULL);
sl@0
  1314
		}
sl@0
  1315
	else
sl@0
  1316
		{
sl@0
  1317
		//Find highest element in window below
sl@0
  1318
		MWsElement* below;
sl@0
  1319
		TInt placedCount = iElements[index-1]->iPlacedElements.Count();
sl@0
  1320
		if (placedCount > 0)
sl@0
  1321
			{
sl@0
  1322
			below = iElements[index-1]->iPlacedElements[placedCount-1].iElement;
sl@0
  1323
			}
sl@0
  1324
		else
sl@0
  1325
			{	//below = above = background surface
sl@0
  1326
			below = iElements[index-1]->iBackgroundElement.iElement;
sl@0
  1327
			}
sl@0
  1328
		returnCode = iScene.InsertSceneElement(element, below);
sl@0
  1329
		}
sl@0
  1330
	
sl@0
  1331
	__ASSERT_DEBUG(returnCode == KErrNone, Panic(EWsPanicSceneErrorIgnored));
sl@0
  1332
sl@0
  1333
	iElements[index]->iBackgroundElement.iElement = element;
sl@0
  1334
	return iElements[index]->iBackgroundElement;
sl@0
  1335
	}
sl@0
  1336
sl@0
  1337
/** Removes the background element of the specified window.
sl@0
  1338
sl@0
  1339
@param aWindow The window to remove the background element from.
sl@0
  1340
@param aWindowClosing ETrue if aWindow is in between closing state.
sl@0
  1341
@return KErrNone on success or a system-wide error code.
sl@0
  1342
*/
sl@0
  1343
TInt CWindowElementSet::ReleaseBackgroundElement(const CWsClientWindow& aWindow, TBool aWindowClosing /*= EFalse*/)
sl@0
  1344
	{
sl@0
  1345
	TInt index;
sl@0
  1346
sl@0
  1347
	TInt err = FindEntry(aWindow, index, aWindowClosing);
sl@0
  1348
	__ASSERT_DEBUG(err != KErrNotFound, Panic(EWsPanicNoWindowElement));
sl@0
  1349
sl@0
  1350
	if (err==KErrNone)
sl@0
  1351
		{
sl@0
  1352
		CWindowElement* winElement=iElements[index];
sl@0
  1353
		if (winElement->iPlacedElements.Count() == 0)
sl@0
  1354
			{	//destroy the window entry
sl@0
  1355
			DestroyWindowElementEntry(index);
sl@0
  1356
			}
sl@0
  1357
		else
sl@0
  1358
			{	// just destroy the background
sl@0
  1359
			if (winElement->iBackgroundElement.iElement)
sl@0
  1360
				{
sl@0
  1361
				winElement->iBackgroundElement.Destroy(iScene, ETrue);
sl@0
  1362
				}
sl@0
  1363
			}
sl@0
  1364
		}
sl@0
  1365
	return err;
sl@0
  1366
	}
sl@0
  1367
sl@0
  1368
/** Destroys the element associated with this set of attributes
sl@0
  1369
sl@0
  1370
@param aScene To allow access to the scene.
sl@0
  1371
@param aUnregister Whether to call unregister on the surface id.
sl@0
  1372
@return ETrue if a element existed to be destroyed, otherwise EFalse.
sl@0
  1373
*/
sl@0
  1374
TBool TAttributes::Destroy(MWsScene& aScene, TBool aUnregister)
sl@0
  1375
	{
sl@0
  1376
	MWsElement* element = iElement;
sl@0
  1377
	if (element)
sl@0
  1378
		{
sl@0
  1379
		TSurfaceId surface = element->ConnectedSurface();
sl@0
  1380
		element->ConnectSurface(TSurfaceId::CreateNullId());
sl@0
  1381
        if (aUnregister)
sl@0
  1382
            {
sl@0
  1383
            aScene.UnregisterSurface(surface);
sl@0
  1384
            }
sl@0
  1385
		aScene.DestroySceneElement(element);
sl@0
  1386
		iElement = NULL;
sl@0
  1387
		}
sl@0
  1388
	return element!=NULL;
sl@0
  1389
	}
sl@0
  1390
sl@0
  1391
/** Destroys all elements associated with the window and removes the window from the element set
sl@0
  1392
sl@0
  1393
@param aWinElementIndex The index of the window in the element set
sl@0
  1394
*/
sl@0
  1395
void CWindowElementSet::DestroyWindowElementEntry(const TInt aWinElementIndex)
sl@0
  1396
	{
sl@0
  1397
	CWindowElement* winElement=iElements[aWinElementIndex];
sl@0
  1398
	const CWsClientWindow& window = winElement->iWindow;
sl@0
  1399
	winElement->iBackgroundElement.Destroy(iScene, ETrue);
sl@0
  1400
	for (TInt placedIndex=0,maxindex=winElement->iPlacedElements.Count();placedIndex<maxindex;placedIndex++)
sl@0
  1401
		{
sl@0
  1402
		winElement->iPlacedElements[placedIndex].Destroy(iScene, EFalse);
sl@0
  1403
		}
sl@0
  1404
	winElement->iPlacedElements.Close();
sl@0
  1405
	delete winElement;
sl@0
  1406
	iElements.Remove(aWinElementIndex);
sl@0
  1407
	window.Redraw()->SetHasElement(EFalse);
sl@0
  1408
	window.Screen()->ElementRemoved();
sl@0
  1409
	}
sl@0
  1410
sl@0
  1411
/** Removes all elements in the associated window.
sl@0
  1412
sl@0
  1413
@param aWindow The window to remove the elements for.
sl@0
  1414
@param aWindowClosing ETrue if aWindow is in between closing state.
sl@0
  1415
@return KErrNone on success, KErrNotFound if there was no windowelement entry.
sl@0
  1416
*/
sl@0
  1417
TInt CWindowElementSet::ReleaseAllElements(const CWsClientWindow& aWindow, TBool aWindowClosing /*= EFalse*/)
sl@0
  1418
	{
sl@0
  1419
	TInt index;
sl@0
  1420
sl@0
  1421
	TInt err = FindEntry(aWindow, index, aWindowClosing);
sl@0
  1422
	__ASSERT_DEBUG(err != KErrNotFound, Panic(EWsPanicNoWindowElement));
sl@0
  1423
	if (err==KErrNone)
sl@0
  1424
		{
sl@0
  1425
		DestroyWindowElementEntry(index);
sl@0
  1426
		}
sl@0
  1427
	return err;
sl@0
  1428
	}
sl@0
  1429
sl@0
  1430
/** For every window element, checks if any of the windows element ids match the one in question.
sl@0
  1431
sl@0
  1432
@param aSurfaceId The surface id to match against.
sl@0
  1433
@return ETrue if a match is found, otherwise EFalse.
sl@0
  1434
sl@0
  1435
@see CWindowElement::SearchDuplicateSurfaceId
sl@0
  1436
*/
sl@0
  1437
TBool CWindowElementSet::SearchDuplicateSurfaceId(const TSurfaceId& aSurfaceId)
sl@0
  1438
	{
sl@0
  1439
	TInt windowElements = iElements.Count();
sl@0
  1440
	for (TInt ii = 0; ii < windowElements; ii++)
sl@0
  1441
		{
sl@0
  1442
		if (iElements[ii]->SearchDuplicateSurfaceId(aSurfaceId))
sl@0
  1443
			{
sl@0
  1444
			return ETrue;
sl@0
  1445
			}
sl@0
  1446
		}
sl@0
  1447
	return EFalse;
sl@0
  1448
	}
sl@0
  1449
sl@0
  1450
/** For a window, sets the opacity for the background surface and if setting to 
sl@0
  1451
0 removes placed surfaces.
sl@0
  1452
sl@0
  1453
@param aWindow The window to perform the function on.
sl@0
  1454
@param aOpacity The opacity to set.
sl@0
  1455
@see CWindowElement::SetElementOpacity
sl@0
  1456
*/
sl@0
  1457
void CWindowElementSet::SetElementOpacity(CWsClientWindow& aWindow, TInt aOpacity)
sl@0
  1458
	{
sl@0
  1459
	WS_ASSERT_DEBUG(aOpacity == 0 || aOpacity == 255,EWsPanicUnexpectedOpacity);
sl@0
  1460
	return FindElement(aWindow)->SetElementOpacity(aOpacity, iScene);
sl@0
  1461
	}
sl@0
  1462
sl@0
  1463
/** Finds all elements within the window that overlap with the region and 
sl@0
  1464
marks them as unassigned.
sl@0
  1465
If any part of these unassigned elements are outside the region, they are also 
sl@0
  1466
marked overlapping and locked config.
sl@0
  1467
sl@0
  1468
@param aRedrawRegion The region in which elements will be unassigned
sl@0
  1469
@param aWindow The window to perform the function on.
sl@0
  1470
@return KErrNone on success or KErrNoMemory if critical calculations could not be made.
sl@0
  1471
@see CWindowElement::UnassignPlacedElements
sl@0
  1472
*/
sl@0
  1473
TInt CWindowElementSet::UnassignPlacedElements(const TRegion& aRedrawRegion, const CWsClientWindow& aWindow, TInt aGcDrawingCount)
sl@0
  1474
	{
sl@0
  1475
	return FindElement(aWindow)->UnassignPlacedElements(aRedrawRegion, aGcDrawingCount);
sl@0
  1476
	}
sl@0
  1477
sl@0
  1478
/** Called to place a element within the window.
sl@0
  1479
It will either create a new element, recycle a element, or extend a element .
sl@0
  1480
sl@0
  1481
@param aPlacedAttributes Returns the attributes of the surface placed.
sl@0
  1482
@param aSurfaceConfiguration The surface configuration for the surface to place.
sl@0
  1483
@param aUserDefinedRegion The user defined clipping of the window.
sl@0
  1484
@param aWindow The window to perform the function on.
sl@0
  1485
@return KErrNone on success or a system-wide error code.
sl@0
  1486
@see CWindowElement::AssignPlacedElement
sl@0
  1487
*/
sl@0
  1488
TInt CWindowElementSet::AssignPlacedElement(
sl@0
  1489
		TPlacedAttributes*& aPlacedAttributes,
sl@0
  1490
		const TSurfaceConfiguration& aSurfaceConfiguration,
sl@0
  1491
		const TRegion& aUserDefinedRegion,
sl@0
  1492
		const CWsClientWindow& aWindow,
sl@0
  1493
		TInt aGcDrawingCount	)
sl@0
  1494
	{
sl@0
  1495
	TInt index;
sl@0
  1496
	TInt error = KErrNone;
sl@0
  1497
	TInt result = FindEntry(aWindow, index);
sl@0
  1498
	TBool insertedElement = EFalse;
sl@0
  1499
	if (result == KErrNotFound)
sl@0
  1500
		{
sl@0
  1501
		CWindowElement* winelement = new CWindowElement(aWindow);
sl@0
  1502
		if (!winelement)
sl@0
  1503
			{
sl@0
  1504
			return KErrNoMemory;		//memory error
sl@0
  1505
			}
sl@0
  1506
		error = iElements.Insert(winelement, index);
sl@0
  1507
		if (error == KErrNone)
sl@0
  1508
			{
sl@0
  1509
			insertedElement = ETrue;
sl@0
  1510
			aWindow.Redraw()->SetHasElement(ETrue);
sl@0
  1511
			STACK_REGION windowRegion = aWindow.Abs();
sl@0
  1512
			if (!windowRegion.CheckError())
sl@0
  1513
				{
sl@0
  1514
				error = iElements[index]->UnassignPlacedElements(windowRegion, aGcDrawingCount);
sl@0
  1515
				}
sl@0
  1516
			else
sl@0
  1517
				{
sl@0
  1518
				error = KErrNoMemory;
sl@0
  1519
				}
sl@0
  1520
			windowRegion.Close();
sl@0
  1521
			}
sl@0
  1522
		else
sl@0
  1523
			{
sl@0
  1524
			delete winelement;
sl@0
  1525
			}
sl@0
  1526
		}
sl@0
  1527
	if (error == KErrNone)
sl@0
  1528
		{
sl@0
  1529
		
sl@0
  1530
		TInt assignReturn = iElements[index]->AssignPlacedElement(aPlacedAttributes, aSurfaceConfiguration,
sl@0
  1531
				aUserDefinedRegion, iScene, aGcDrawingCount);
sl@0
  1532
		error = assignReturn;	//return assign flags
sl@0
  1533
		}
sl@0
  1534
	if (error < KErrNone && insertedElement)
sl@0
  1535
		{	//remove this element that has just been created
sl@0
  1536
		aWindow.Redraw()->SetHasElement(EFalse);
sl@0
  1537
		iElements.Remove(index);
sl@0
  1538
		}
sl@0
  1539
	return error;
sl@0
  1540
	}
sl@0
  1541
sl@0
  1542
/** Marks all elements that have been assigned since the unassign as drawn over.
sl@0
  1543
sl@0
  1544
@param aWindow The window to perform the function on.
sl@0
  1545
@see CWindowElement::FlagAssignedElementsDrawnOver
sl@0
  1546
**/
sl@0
  1547
//void CWindowElementSet::FlagAssignedElementsDrawnOver(const CWsClientWindow& aWindow)
sl@0
  1548
//	{
sl@0
  1549
//	FindElement(aWindow)->FlagAssignedElementsDrawnOver();
sl@0
  1550
//	}
sl@0
  1551
sl@0
  1552
/** Called at the end of a redraw to set cached changed to elements, and
sl@0
  1553
removed elements that are still unassigned.
sl@0
  1554
sl@0
  1555
@param aWindow The window to perform the function on.
sl@0
  1556
@return Flags to describe what extra work needed doing.
sl@0
  1557
@see CWindowElement::CleanUpPlacedElements
sl@0
  1558
*/
sl@0
  1559
TInt CWindowElementSet::CleanUpPlacedElements(const CWsClientWindow& aWindow, TInt aGcDrawingCount)
sl@0
  1560
	{
sl@0
  1561
	TInt index;
sl@0
  1562
	TInt error = FindEntry(aWindow,index);
sl@0
  1563
	if (error < 0)
sl@0
  1564
		{
sl@0
  1565
		WS_ASSERT_DEBUG(0,EWsPanicNoWindowElement);
sl@0
  1566
		return error;
sl@0
  1567
		}
sl@0
  1568
	TInt returnFlags = iElements[index]->CleanUpPlacedElements(iScene, aGcDrawingCount);
sl@0
  1569
sl@0
  1570
	if (!iElements[index]->iBackgroundElement.iElement && iElements[index]->iPlacedElements.Count() == 0)
sl@0
  1571
		{	//remove window entry
sl@0
  1572
		DestroyWindowElementEntry(index);
sl@0
  1573
		}
sl@0
  1574
	else
sl@0
  1575
		{
sl@0
  1576
		if (returnFlags&CWindowElement::EResort)
sl@0
  1577
			{	//need to sort elements
sl@0
  1578
			//find front most visible element in window behind, invoke sort by z order
sl@0
  1579
			if (index == 0)
sl@0
  1580
				{	//this element is backmost window
sl@0
  1581
				iElements[index]->MoveToAboveGivenElement(iScene,NULL);
sl@0
  1582
				}
sl@0
  1583
			else
sl@0
  1584
				{	//place infront of highest element behind
sl@0
  1585
				if (iElements[index-1]->iPlacedElements.Count() == 0)
sl@0
  1586
					{	//top element of element behind must be the background element
sl@0
  1587
					iElements[index]->MoveToAboveGivenElement(iScene,iElements[index-1]->iBackgroundElement.iElement);				
sl@0
  1588
					}
sl@0
  1589
				else
sl@0
  1590
					{	//top element of element behind must be highest placed element
sl@0
  1591
					TInt placedCount = iElements[index-1]->iPlacedElements.Count();
sl@0
  1592
					iElements[index]->MoveToAboveGivenElement(iScene,iElements[index-1]->iPlacedElements[placedCount-1].iElement);
sl@0
  1593
					}
sl@0
  1594
				}
sl@0
  1595
			}
sl@0
  1596
		}
sl@0
  1597
	if (returnFlags&CWindowElement::EFailed)
sl@0
  1598
		{	//visible elements may have been deleted, error to suggest a full window redraw
sl@0
  1599
		return KErrGeneral;
sl@0
  1600
		}
sl@0
  1601
	return returnFlags;
sl@0
  1602
	}
sl@0
  1603
sl@0
  1604
/** Counts how many elements there are in the set.
sl@0
  1605
sl@0
  1606
@return The number of elements in the set.
sl@0
  1607
*/
sl@0
  1608
TInt CWindowElementSet::Count() const
sl@0
  1609
	{
sl@0
  1610
	return iElements.Count();
sl@0
  1611
	}
sl@0
  1612
sl@0
  1613
/** Returns the background attributes for the specified window
sl@0
  1614
sl@0
  1615
@param aWindow The window to perform the function on.
sl@0
  1616
@return The background surface attributes.  If the window has no elementset entry, returns NULL.
sl@0
  1617
*/
sl@0
  1618
TBackgroundAttributes* CWindowElementSet::FindBackgroundElement(const CWsClientWindow& aWindow)
sl@0
  1619
	{
sl@0
  1620
	TInt index;
sl@0
  1621
	TInt err = FindEntry(aWindow, index);
sl@0
  1622
sl@0
  1623
	if (err != KErrNotFound)
sl@0
  1624
		{
sl@0
  1625
		return &(iElements[index]->iBackgroundElement);
sl@0
  1626
		}
sl@0
  1627
	return NULL;
sl@0
  1628
	}
sl@0
  1629
sl@0
  1630
/**	Returns the contents of the element data associated with the input window.
sl@0
  1631
If this method is successful, then neither pointer will be NULL.
sl@0
  1632
sl@0
  1633
@return standard symbian error code.
sl@0
  1634
@param	aWindow	window to find
sl@0
  1635
@param	aBackAttr	backgroud surface attributes associated with the window
sl@0
  1636
@param	aPlacedAttr	array of placed surface attributes associated with the window.
sl@0
  1637
*/
sl@0
  1638
TInt CWindowElementSet::FindElements(	CWsClientWindow const &aWindow, 
sl@0
  1639
									TBackgroundAttributes const * & aBackAttr, 
sl@0
  1640
									RArray<class TPlacedAttributes> const * &	aPlacedAttr	)
sl@0
  1641
	{
sl@0
  1642
	TInt index;
sl@0
  1643
sl@0
  1644
	TInt err = FindEntry(aWindow, index);
sl@0
  1645
//	__ASSERT_DEBUG(err != KErrNotFound, Panic(EWsPanicNoWindowElement));
sl@0
  1646
sl@0
  1647
//	return *iElements[index].iElement;
sl@0
  1648
	if (err >= KErrNone)
sl@0
  1649
		{
sl@0
  1650
		aBackAttr=&iElements[index]->iBackgroundElement;
sl@0
  1651
		aPlacedAttr=&iElements[index]->iPlacedElements;
sl@0
  1652
		}
sl@0
  1653
	else
sl@0
  1654
		{
sl@0
  1655
		aBackAttr=NULL;
sl@0
  1656
		aPlacedAttr=NULL;
sl@0
  1657
		}
sl@0
  1658
	return err;
sl@0
  1659
	
sl@0
  1660
	}
sl@0
  1661
sl@0
  1662
/** Registers the surface with the compositor.
sl@0
  1663
@param aSurface The surface id to register.
sl@0
  1664
@return  KErrNone if successful, KErrNoMemory if registration fails due to low
sl@0
  1665
memory, KErrNotSupported if the surface is not compatible with
sl@0
  1666
this compositor or KErrBadHandle if the given surface ID does not
sl@0
  1667
represent a valid surface. KErrArgument is returned if the
sl@0
  1668
surface does not have both dimensions less than 32 767 pixels.
sl@0
  1669
*/
sl@0
  1670
TInt CWindowElementSet::RegisterSurface(const TSurfaceId& aSurface)
sl@0
  1671
    {
sl@0
  1672
    
sl@0
  1673
    return iScene.RegisterSurface(aSurface);
sl@0
  1674
    }
sl@0
  1675
sl@0
  1676
/** Unregisters the surface with the compositor.
sl@0
  1677
sl@0
  1678
@param aSurface The surface id to register.
sl@0
  1679
@return  KErrNone if successful. KErrInUse if the surface is
sl@0
  1680
used by a layer or layers. KErrBadHandle if the surface
sl@0
  1681
is not currently registered. KErrArgument if
sl@0
  1682
the surface ID is a NULL ID.
sl@0
  1683
*/
sl@0
  1684
sl@0
  1685
void CWindowElementSet::UnregisterSurface(const TSurfaceId& aSurface)
sl@0
  1686
    {
sl@0
  1687
    
sl@0
  1688
    TInt returnCode = iScene.UnregisterSurface(aSurface);
sl@0
  1689
    __ASSERT_DEBUG((returnCode==KErrNone || returnCode==KErrInUse || returnCode==KErrBadHandle), Panic(EWsPanicSceneErrorIgnored));
sl@0
  1690
    }
sl@0
  1691
sl@0
  1692
/** Sorts the array elements into the same order as the windows are in the
sl@0
  1693
hierarchy. Use after window hierarchy has been modified, to update the scene
sl@0
  1694
order to match.
sl@0
  1695
@return EFalse if element order unchanged, ETrue if order may have changed.
sl@0
  1696
*/
sl@0
  1697
TBool CWindowElementSet::SortByZOrder()
sl@0
  1698
	{
sl@0
  1699
	if (iElements.Count() < 2)
sl@0
  1700
		{
sl@0
  1701
		// Early out for the very common cases where there are zero or one
sl@0
  1702
		// elements, which cannot therefore be out of order.
sl@0
  1703
		return EFalse;
sl@0
  1704
		}
sl@0
  1705
sl@0
  1706
	// The approach being used is to first just sort the array, then update the 
sl@0
  1707
	// scene order afterwards. This is simple to code and since there are not
sl@0
  1708
	// expected to be many elements, it should be perfectly good enough.
sl@0
  1709
sl@0
  1710
	TLinearOrder<CWindowElement> order(WindowOrder);
sl@0
  1711
	iElements.Sort(order);
sl@0
  1712
sl@0
  1713
	TBool orderChanged = EFalse;
sl@0
  1714
	MWsElement* below;
sl@0
  1715
	TInt elementCount = iElements.Count();
sl@0
  1716
	for (TInt index = 0; index < elementCount; index++)
sl@0
  1717
		{
sl@0
  1718
		if (index == 0)
sl@0
  1719
			{
sl@0
  1720
			below = NULL;
sl@0
  1721
			}
sl@0
  1722
		else
sl@0
  1723
			{
sl@0
  1724
			//Find highest element in window below
sl@0
  1725
			TInt placedCount = iElements[index-1]->iPlacedElements.Count();
sl@0
  1726
			if (placedCount > 0)
sl@0
  1727
				{
sl@0
  1728
				below = iElements[index-1]->iPlacedElements[placedCount-1].iElement;
sl@0
  1729
				}
sl@0
  1730
			else
sl@0
  1731
				{	//below = above = background surface
sl@0
  1732
				below = iElements[index-1]->iBackgroundElement.iElement;
sl@0
  1733
				}
sl@0
  1734
			}
sl@0
  1735
		//Find lowest element in above window element
sl@0
  1736
		MWsElement* above;
sl@0
  1737
		if (iElements[index]->iBackgroundElement.iElement != NULL)
sl@0
  1738
			{	//use background element
sl@0
  1739
			above = iElements[index]->iBackgroundElement.iElement;	//background element will be bottom
sl@0
  1740
			}
sl@0
  1741
		else	//use bottom element of placed surfaces
sl@0
  1742
			{
sl@0
  1743
			//above = iElements[index]->iPlacedElements[iElements[index]->iPlacedElements.Count()-1].iElement;	//first or last placed element is bottom?
sl@0
  1744
			above = iElements[index]->iPlacedElements[0].iElement;
sl@0
  1745
			}
sl@0
  1746
		if (above->ElementBelow() != below)
sl@0
  1747
			{
sl@0
  1748
			//CALL below window element function to move all elements above 'below'
sl@0
  1749
			iElements[index]->MoveToAboveGivenElement(iScene, below);
sl@0
  1750
	        orderChanged = ETrue;
sl@0
  1751
			}
sl@0
  1752
		}
sl@0
  1753
	return orderChanged;
sl@0
  1754
	}
sl@0
  1755
sl@0
  1756
/** Processes the specified windows placed elements, for when windows visibility is changed.
sl@0
  1757
If the placed elements are no longer visible, they are removed.
sl@0
  1758
sl@0
  1759
@param aWindow The window to call the function on.
sl@0
  1760
@return Positive if any elements have been removed, zero if not, or errorcode. 
sl@0
  1761
@see CWindowElement::SetVisibleRegion
sl@0
  1762
*/
sl@0
  1763
TInt CWindowElementSet::SetVisibleRegion(CWsClientWindow& aWindow)
sl@0
  1764
	{
sl@0
  1765
	TInt index;
sl@0
  1766
	TInt find = FindEntry(aWindow,index);
sl@0
  1767
	WS_ASSERT_DEBUG(find>=KErrNone,EWsPanicNoWindowElement);
sl@0
  1768
	TBool ret = iElements[index]->SetVisibleRegion(aWindow.VisibleRegion(), iScene);
sl@0
  1769
sl@0
  1770
	if (!iElements[index]->iBackgroundElement.iElement && iElements[index]->iPlacedElements.Count() == 0)
sl@0
  1771
		{
sl@0
  1772
		DestroyWindowElementEntry(index);
sl@0
  1773
		}
sl@0
  1774
	return ret;
sl@0
  1775
	}
sl@0
  1776
sl@0
  1777
/** Updates the specified windows elements extent, either from a window movement or a change in window size
sl@0
  1778
sl@0
  1779
@param aWindow The window to call the function on.
sl@0
  1780
@param aOffset The movement of the window.  If NULL then the window has changed size.
sl@0
  1781
@see CWindowElement::UpdateElementExtent
sl@0
  1782
*/
sl@0
  1783
void CWindowElementSet::UpdateElementExtent(const CWsClientWindow& aWindow, const TPoint* aOffset)
sl@0
  1784
	{
sl@0
  1785
	FindElement(aWindow)->UpdateElementExtent(aOffset);
sl@0
  1786
	}
sl@0
  1787
sl@0
  1788
/** Method to fill in a TSurfaceConfiguration from a scene element.
sl@0
  1789
sl@0
  1790
@param aConfiguration Surface configuration to fill in.
sl@0
  1791
@param aElement Element to get information from.
sl@0
  1792
@return Once multiple versions of TSurfaceConfiguration are available, KErrNotSupported if configuration supplied is too small.
sl@0
  1793
*/
sl@0
  1794
TInt CWindowElementSet::GetConfiguration(TSurfaceConfiguration& aConfiguration,MWsElement& aElement)
sl@0
  1795
	{
sl@0
  1796
	TSurfaceId tempSurfaceId = aElement.ConnectedSurface();
sl@0
  1797
	aConfiguration.SetSurfaceId(tempSurfaceId);
sl@0
  1798
	
sl@0
  1799
	//Convert and copy orientation
sl@0
  1800
	aConfiguration.SetOrientation(ElementToGcRotation(aElement.SourceRotation()));
sl@0
  1801
	
sl@0
  1802
	//Convert and copy flip
sl@0
  1803
	TBool flip = aElement.SourceFlipping();
sl@0
  1804
	aConfiguration.SetFlip(flip);
sl@0
  1805
sl@0
  1806
	//Convert and copy viewport
sl@0
  1807
	TRect tempViewport;
sl@0
  1808
	aElement.GetSourceRectangle(tempViewport);
sl@0
  1809
	aConfiguration.SetViewport(tempViewport);
sl@0
  1810
	TRect tempExtent;
sl@0
  1811
	aElement.GetDestinationRectangle(tempExtent);
sl@0
  1812
	aConfiguration.SetExtent(tempExtent);
sl@0
  1813
	return KErrNone;	//Could fail if there are multiple versions of TSurfaceConfiguration
sl@0
  1814
	}
sl@0
  1815
sl@0
  1816
/** Returns the window element entry for the specified window.
sl@0
  1817
sl@0
  1818
@param aWindow The window to call the function on.
sl@0
  1819
*/
sl@0
  1820
CWindowElement* CWindowElementSet::FindElement(const CWsClientWindow& aWindow) const
sl@0
  1821
	{
sl@0
  1822
	TInt index;
sl@0
  1823
	TInt error = FindEntry(aWindow,index);
sl@0
  1824
	if (error == KErrNone)
sl@0
  1825
		return iElements[index];
sl@0
  1826
	WS_ASSERT_DEBUG(EFalse,EWsPanicNoWindowElement);
sl@0
  1827
	return NULL;
sl@0
  1828
	}
sl@0
  1829
sl@0
  1830
/** Creates a set of window-element pairs for this scene.
sl@0
  1831
sl@0
  1832
@param aScene To allow access to the scene.
sl@0
  1833
@param aComposer To allow access to the composer.
sl@0
  1834
*/
sl@0
  1835
CWindowElementSet::CWindowElementSet(MWsScene& aScene) :
sl@0
  1836
	iScene(aScene)
sl@0
  1837
	{}
sl@0
  1838
sl@0
  1839
/** Searches for the entry in iElements with the given window
sl@0
  1840
sl@0
  1841
@param aWindow The window to find the entry of / where it should be inserted.
sl@0
  1842
@param aIndex aIndex is set to entry found or the insertion point, respectively.
sl@0
  1843
@param aLinearSearch ETrue if a linear search of the window element set is required.
sl@0
  1844
@return KErrNone if found or KErrNotFound.
sl@0
  1845
*/
sl@0
  1846
TInt CWindowElementSet::FindEntry(const CWsClientWindow& aWindow, TInt& aIndex, TBool aLinearSearch /*= EFalse*/) const
sl@0
  1847
	{
sl@0
  1848
	CWindowElement winelement(aWindow);
sl@0
  1849
	
sl@0
  1850
	// The array order makes use of the parent pointer, which gets reset during
sl@0
  1851
	// window shutdown, so if it is clear fall back to a linear search.
sl@0
  1852
	if (!aWindow.BaseParent() || aLinearSearch)
sl@0
  1853
		{
sl@0
  1854
		TIdentityRelation<CWindowElement> match(WindowMatch);
sl@0
  1855
		
sl@0
  1856
		aIndex = iElements.Find(&winelement, match);
sl@0
  1857
		return (aIndex == KErrNotFound) ? KErrNotFound : KErrNone;
sl@0
  1858
		}
sl@0
  1859
	else
sl@0
  1860
		{
sl@0
  1861
		TLinearOrder<CWindowElement> order(WindowOrder);
sl@0
  1862
		return iElements.FindInOrder(&winelement, aIndex, order);
sl@0
  1863
		}
sl@0
  1864
	}
sl@0
  1865
sl@0
  1866
/** Used to find an entry in the set when order cannot be used. 
sl@0
  1867
sl@0
  1868
@param aFirst First windowelement to compare.
sl@0
  1869
@param aSecond Second windowelement to compare.
sl@0
  1870
@return ETrue if the entries are the same, and EFalse if they are not.
sl@0
  1871
*/
sl@0
  1872
TBool CWindowElementSet::WindowMatch(const CWindowElement& aFirst, const CWindowElement& aSecond)
sl@0
  1873
	{
sl@0
  1874
	return (&aFirst.iWindow == &aSecond.iWindow);
sl@0
  1875
	}
sl@0
  1876
sl@0
  1877
sl@0
  1878
/** Used to determine the order of entries in the set.
sl@0
  1879
sl@0
  1880
@param aFirst First windowelement to compare.
sl@0
  1881
@param aSecond Second windowelement to compare.
sl@0
  1882
@return zero if the entries are the same, a negative value if 
sl@0
  1883
aFirst is behind aSecond or a positive value if aFirst is in front of aSecond.
sl@0
  1884
*/
sl@0
  1885
TInt CWindowElementSet::WindowOrder(const CWindowElement& aFirst, const CWindowElement& aSecond)
sl@0
  1886
	{
sl@0
  1887
	TInt result = 0;
sl@0
  1888
sl@0
  1889
	if (&aFirst.iWindow != &aSecond.iWindow)
sl@0
  1890
		{
sl@0
  1891
		result = aFirst.iWindow.IsInfrontOf(&aSecond.iWindow) ? 1 : -1;
sl@0
  1892
		}
sl@0
  1893
sl@0
  1894
	return result;
sl@0
  1895
	}
sl@0
  1896
sl@0
  1897
sl@0
  1898
/**	Re-sends the extents for all the elements (in this window) to the scene  	
sl@0
  1899
 * 	This allows the renderstage to re-scale those element extents
sl@0
  1900
 * 
sl@0
  1901
 * 
sl@0
  1902
 **/
sl@0
  1903
void CWindowElement::ResubmitAllElementExtents()
sl@0
  1904
	{
sl@0
  1905
	if (MWsElement* element=iBackgroundElement.iElement)
sl@0
  1906
		{
sl@0
  1907
		TRect extent(TRect::EUninitialized);
sl@0
  1908
		element->GetDestinationRectangle(extent);
sl@0
  1909
		element->SetDestinationRectangle(extent);
sl@0
  1910
		}
sl@0
  1911
	TInt elementCount = iPlacedElements.Count();
sl@0
  1912
	for (TInt index = 0; index < elementCount; index++)
sl@0
  1913
		{
sl@0
  1914
		if (MWsElement* element=iPlacedElements[index].iElement)
sl@0
  1915
			{
sl@0
  1916
			TRect extent(TRect::EUninitialized);
sl@0
  1917
			element->GetDestinationRectangle(extent);
sl@0
  1918
			element->SetDestinationRectangle(extent);
sl@0
  1919
			}
sl@0
  1920
		}
sl@0
  1921
	}
sl@0
  1922
sl@0
  1923
/**	Re-sends the extents for all the elements (in all the windows) to the scene  	
sl@0
  1924
 * 	This allows the renderstage to re-scale those element extents
sl@0
  1925
 * 
sl@0
  1926
 * 
sl@0
  1927
 **/
sl@0
  1928
void CWindowElementSet::ResubmitAllElementExtents()
sl@0
  1929
	{
sl@0
  1930
	TInt elementCount = iElements.Count();
sl@0
  1931
	for (TInt index = 0; index < elementCount; index++)
sl@0
  1932
		{
sl@0
  1933
		iElements[index]->ResubmitAllElementExtents();
sl@0
  1934
		}
sl@0
  1935
	}
sl@0
  1936
sl@0
  1937
//
sl@0
  1938
// Debug functions
sl@0
  1939
//
sl@0
  1940
sl@0
  1941
/** Returns background attributes for the specified window index.
sl@0
  1942
For use with debug client interface.
sl@0
  1943
sl@0
  1944
@param aWin Window index to get the surface attributes from.
sl@0
  1945
@return Background attributes for the specified window.  NULL if the window index is invalid.
sl@0
  1946
*/
sl@0
  1947
const TBackgroundAttributes* CWindowElementSet::DebugBackgroundAt(TUint aWin)const
sl@0
  1948
	{
sl@0
  1949
	if (aWin>=Count())
sl@0
  1950
		return NULL;
sl@0
  1951
	return &iElements[aWin]->iBackgroundElement;
sl@0
  1952
	}
sl@0
  1953
sl@0
  1954
/** Returns the client window for the specified window index.
sl@0
  1955
For use with debug client interface.
sl@0
  1956
sl@0
  1957
@param aWin Window index to get the client window from.
sl@0
  1958
@return Client window for the specified window.  NULL if the window index is invalid.
sl@0
  1959
*/
sl@0
  1960
const CWsClientWindow* CWindowElementSet::DebugWindowAt(TUint aWin)const
sl@0
  1961
	{
sl@0
  1962
	if (aWin>=Count())
sl@0
  1963
		return NULL;
sl@0
  1964
	return &iElements[aWin]->iWindow;
sl@0
  1965
	}
sl@0
  1966
sl@0
  1967
/** Returns how many placed surfaces are on the specified window
sl@0
  1968
For use with debug client interface.
sl@0
  1969
sl@0
  1970
@param aWin Window index to get the client window from.
sl@0
  1971
@return Amount of placed surfaces for the specified window index
sl@0
  1972
*/
sl@0
  1973
TInt CWindowElementSet::DebugPlacedCountAt(TUint aWin)const
sl@0
  1974
	{
sl@0
  1975
	if (aWin>=Count())
sl@0
  1976
		return -1;
sl@0
  1977
	return iElements[aWin]->iPlacedElements.Count();
sl@0
  1978
	}
sl@0
  1979
sl@0
  1980
/** Returns the placed attributes for the specified placed surface index in the
sl@0
  1981
specified window index.
sl@0
  1982
For use with debug client interface.
sl@0
  1983
sl@0
  1984
@param aWin Window index to get the client window from.
sl@0
  1985
@param aPlace Placed surface index to get the placed surface attributes
sl@0
  1986
@return Placed surface attributes.  NULL if either index was invalid.
sl@0
  1987
*/
sl@0
  1988
const TPlacedAttributes* CWindowElementSet::DebugPlacedAt(TUint aWin,TUint aPlace)const
sl@0
  1989
	{
sl@0
  1990
	if (aWin>=Count())
sl@0
  1991
		return NULL;
sl@0
  1992
	if (aPlace>=iElements[aWin]->iPlacedElements.Count())
sl@0
  1993
		return NULL;
sl@0
  1994
	return &iElements[aWin]->iPlacedElements[aPlace];
sl@0
  1995
	}
sl@0
  1996
sl@0
  1997
MWsElement* CWindowElementSet::GetElementFromWindow(const CWsClientWindow& aWindow) const
sl@0
  1998
    {
sl@0
  1999
    CWindowElement* windowElement = FindElement(aWindow);
sl@0
  2000
    if (windowElement)
sl@0
  2001
        return windowElement->Element();
sl@0
  2002
    else 
sl@0
  2003
        return NULL; 
sl@0
  2004
    }