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 + }