First public contribution.
1 // Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
19 #include "windowelementset.h"
20 #include "regionextend.h"
21 #include <graphics/wsscene.h>
22 #include <graphics/wsgraphicdrawer.h>
28 #ifndef KCacheListGranularity
29 enum {KCacheListGranularity = 8};
31 #ifndef KCacheExtensionReserve
32 enum {KCacheExtensionReserve = 1};
37 /** Element to gc rotation function
39 LOCAL_C inline CFbsBitGc::TGraphicsOrientation ElementToGcRotation(MWsElement::TElementRotation aElementRotation)
41 CFbsBitGc::TGraphicsOrientation gcRotation = CFbsBitGc::EGraphicsOrientationNormal;
43 switch (aElementRotation)
45 case MWsElement::EElementAntiClockwise90:
46 gcRotation = CFbsBitGc::EGraphicsOrientationRotated90;
48 case MWsElement::EElementAntiClockwise180:
49 gcRotation = CFbsBitGc::EGraphicsOrientationRotated180;
51 case MWsElement::EElementAntiClockwise270:
52 gcRotation = CFbsBitGc::EGraphicsOrientationRotated270;
64 CWindowElement::CWindowElement(const CWsClientWindow& aWindow) :
65 iWindow(aWindow), iCache(KCacheListGranularity), iHighestReusedIndex(-1)
69 /** Finds all elements within the window that overlap with the region and
70 marks them as unassigned.
71 If any part of these unassigned elements are outside the region, they are also
72 marked overlapping and locked config.
74 @param aRegion The region that is being redrawn.
75 @return KErrNone on success or KErrNoMemory if critical calculations could not be made.
77 @see CWindowElementSet::UnassignPlacedElements
79 TInt CWindowElement::UnassignPlacedElements(const TRegion& aRegion,TInt aGcDrawingCount)
81 iFlags = 0; //reset flags
82 iStartGcDrawingCount = aGcDrawingCount; //Detect any drawing between calls
83 iBackgroundElement.SetDrawnOverLast(iBackgroundElement.DrawnOver());
84 iNonRedrawVisibleRegion.Clear();
85 iNonRedrawVisibleRegion.Copy(const_cast<CWsClientWindow&>(iWindow).VisibleRegion());
86 iNonRedrawVisibleRegion.SubRegion(aRegion);
87 if (iNonRedrawVisibleRegion.CheckError())
88 { //memory error. Still needs to cope if whole window is being redrawn
89 iNonRedrawVisibleRegion.Clear();
90 TInt fullWindowRedraw = TRegionExtend::Cast(aRegion).TestDifference(const_cast<CWsClientWindow&>(iWindow).VisibleRegion());
91 if (fullWindowRedraw != 0)
92 { //not a full window redraw, return immediately
96 TInt placedElements = iPlacedElements.Count();
97 TInt unassignedCount = 0;
99 TBool cacheError = EFalse;
101 STACK_REGION tempSurfaceBaseArea;
102 if (iCache.Reserve(placedElements + KCacheExtensionReserve) < KErrNone) //quite often there may need to be room for more
103 { //failed to reserve space
109 for (ii=0; ii<placedElements; ii++)
112 { //cache is ok up to this point but does not contain cache for this element yet
114 if (iCache.Append(newRegion) < KErrNone)
121 iPlacedElements[ii].iElement->GetDestinationRectangle(tempExtent);
122 TInt regionReturn = TRegionExtend::Cast(aRegion).TestDifference(tempExtent);
123 iPlacedElements[ii].SetDrawnOverLast(iPlacedElements[ii].DrawnOver()); //allows detection of change of state
124 if (!(regionReturn&TRegionExtend::ENoIntersect))
125 { //redraw region intersects with element
126 iPlacedElements[ii].SetUnassigned();
127 iPlacedElements[ii].SetDrawnOver(EFalse);
130 if (!(regionReturn&TRegionExtend::EAdd))
131 { //surface is entirely within the redraw region
132 iPlacedElements[ii].SetOverlapping(EFalse);
133 iPlacedElements[ii].SetLockedConfig(EFalse);
136 TRect destinationRect;
137 tempSurfaceBaseArea.Clear();
138 iPlacedElements[ii].iElement->GetDestinationRectangle(destinationRect);
139 tempSurfaceBaseArea.AddRect(destinationRect);
140 if (!tempSurfaceBaseArea.CheckError())
142 tempSurfaceBaseArea.Offset(tempExtent.iTl); //cache needs to be in absolute coords
143 iCache[ii].Copy(tempSurfaceBaseArea);
145 if (tempSurfaceBaseArea.CheckError() || iCache[ii].CheckError())
146 { //cache is no good from this point onwards
155 TRect destinationRect;
156 tempSurfaceBaseArea.Clear();
157 iPlacedElements[ii].iElement->GetDestinationRectangle(destinationRect);
158 tempSurfaceBaseArea.AddRect(destinationRect);
159 if (tempSurfaceBaseArea.CheckError())
160 { //critical failure, clear cache and return error
161 for (ii=iCache.Count(); ii>0; ii--)
163 iCache[ii-1].Close();
167 iNonRedrawVisibleRegion.Close();
168 tempSurfaceBaseArea.Close();
173 tempSurfaceBaseArea.Offset(tempExtent.iTl); //change to absolute coordinates
174 regionReturn = TRegionExtend::Cast(iNonRedrawVisibleRegion ).TestDifference(tempSurfaceBaseArea);
175 if (regionReturn&TRegionExtend::ENoIntersect)
176 { //element base area entirely inside redraw region
177 iPlacedElements[ii].SetOverlapping(EFalse);
178 iPlacedElements[ii].SetLockedConfig(EFalse);
181 { //element base area is at least partly within non redraw visible region
182 iPlacedElements[ii].SetOverlapping(ETrue);
183 iPlacedElements[ii].SetLockedConfig(ETrue);
187 iCache[ii].Copy(tempSurfaceBaseArea);
188 if (iCache[ii].CheckError())
189 { //cache is no good from this point onwards
199 { //element does not intersect with redraw region
200 iPlacedElements[ii].SetUnassigned(EFalse);
203 tempSurfaceBaseArea.Close();
205 if (iRemovedSurfaces.Reserve(unassignedCount) == KErrNone)
207 iRemovedSurfacesValid = ETrue;
211 iRemovedSurfacesValid = EFalse;
216 /** Marks all elements that have been assigned since the unassign as drawn over.
218 @see CWindowElementSet::FlagAssignedElementsDrawnOver
220 void CWindowElement::FlagAssignedElementsDrawnOver(TInt aGcDrawingCount)
222 iStartGcDrawingCount = aGcDrawingCount;
223 if (iBackgroundElement.iElement)
225 iBackgroundElement.SetDrawnOver();
227 TInt placedElementCount = iPlacedElements.Count();
228 for (TInt ii = 0; ii < placedElementCount; ii++)
230 if(!iPlacedElements[ii].Unassigned() && iPlacedElements[ii].LockedConfig())
232 iPlacedElements[ii].SetDrawnOver();
234 else if(iPlacedElements[ii].Unassigned() && iPlacedElements[ii].LockedConfig() &&
235 !iPlacedElements[ii].ChangedClip() && iPlacedElements[ii].Overlapping())
237 iPlacedElements[ii].SetDrawnOver();
242 /** Checks whether a elements configuration is the same as that of the surface configuration, taking
243 into account the stateflags (TPlacedAttributes) passed in.
244 It must match surfaceId and extent.
245 If it is a locked config it will need to match viewport and orientation as well.
247 @param aElement Element to compare against.
248 @param aSurfaceConfiguration Configuration to compare.
249 @param aStateFlags The TPlacedAttributes flags for the element.
250 @return ETrue if a match, otherwise EFalse.
252 TBool CWindowElement::IsElementAMatch(MWsElement& aElement, const TSurfaceConfiguration& aSurfaceConfiguration,
255 TBool match = EFalse;
257 TRect tempElementExtent;
259 aElement.GetDestinationRectangle(tempElementExtent);
260 aSurfaceConfiguration.GetExtent(tempCRPExtent);
261 if (tempElementExtent == tempCRPExtent)
263 TSurfaceId tempElementSurfaceId;
264 TSurfaceId tempCRPSurfaceId;
265 aSurfaceConfiguration.GetSurfaceId(tempElementSurfaceId);
266 tempCRPSurfaceId = aElement.ConnectedSurface();
267 if (tempElementSurfaceId == tempCRPSurfaceId)
270 if (aStateFlags&TPlacedAttributes::ELockedConfig)
273 TBool orientationOK = EFalse;
274 TRect tempCRPViewport;
275 aSurfaceConfiguration.GetViewport(tempCRPViewport);
276 //if explicit viewport flagged
277 TRect tempElementViewport;
278 aElement.GetSourceRectangle(tempElementViewport);
279 if (tempElementViewport == tempCRPViewport)
281 CFbsBitGc::TGraphicsOrientation tempElementOrientation;
282 MWsElement::TElementRotation tempCRPRotation;
283 CFbsBitGc::TGraphicsOrientation tempCRPOrientation;
284 tempElementOrientation = aSurfaceConfiguration.Orientation();
285 tempCRPRotation = aElement.SourceRotation();
286 tempCRPOrientation = ElementToGcRotation(tempCRPRotation); //convert to common type
287 if (tempElementOrientation == tempCRPOrientation)
289 orientationOK = ETrue;
294 TBool tempElementFlip = aSurfaceConfiguration.Flip();
295 TBool tempCRPFlip = aElement.SourceFlipping();
296 if(tempElementFlip == tempCRPFlip)
307 /** Sets the surfaceconfiguration to the element.
309 @param aElement The element to set.
310 @param aSurfaceConfiguration The configuration to set to the element.
311 @param aLimitedSet Set all values if EFalse, otherwise only set viewport and orientation.
312 @return KErrNone on success, error from compositor if it could not set the surface id.
314 TInt CWindowElement::SetElement(MWsElement& aElement, const TSurfaceConfiguration& aSurfaceConfiguration, TBool aLimitedSet)
315 { //aLimitedSet - EFalse = set all . ETrue = set viewport and orientation only
319 TSurfaceId newSurfaceId;
320 aSurfaceConfiguration.GetSurfaceId(newSurfaceId);
321 TInt error = aElement.ConnectSurface(newSurfaceId);
322 if (error != KErrNone)
329 aSurfaceConfiguration.GetExtent(newExtent);
330 aElement.SetDestinationRectangle(newExtent);
335 aSurfaceConfiguration.GetViewport(newViewport);
336 if (newViewport.iTl.iX < 0 || newViewport.iTl.iY < 0)
341 if (!newViewport.IsEmpty())
342 { //need to check if viewport is valid size
343 TInt error = aElement.SetSourceRectangle(newViewport);
344 if (error != KErrNone)
351 //Note for compatibility with Oghma:
352 //This method does not properly error check or return KErrArgument,
353 //as that would propagate a client panic in SetBackgroundSurface which did not previously occur.
354 aElement.SetSourceRotation(GcToElementRotation(aSurfaceConfiguration.Orientation()));
355 // Set or clear flip if the element flags are changing
356 if ( (!aSurfaceConfiguration.Flip()) != (!aElement.SourceFlipping()) )
358 aElement.SetSourceFlipping(!aElement.SourceFlipping());
363 /** Attempts to insert a region cache in to the the windows iCache.
364 If it fails, it will remove all cache beyond its index as it is now out of step.
366 @param aIndexToInsert The index of where to insert the cache.
367 @return ETrue on success, else EFalse.
370 TBool CWindowElement::InsertCache(TInt aIndexToInsert)
373 TInt cacheInsertError = iCache.Insert(newRegion, aIndexToInsert);
374 if (cacheInsertError == KErrNone)
377 return ETrue; //inserted cache
379 if (iCache.Count() > aIndexToInsert)
380 { //if insert failed and there is more cache above where its trying to insert
381 iCache.Remove(iCache.Count()-1);
382 cacheInsertError = iCache.Insert(newRegion, aIndexToInsert);
384 if (cacheInsertError == KErrNone)
387 return ETrue; //inserted cache
391 //cant insert this cache, there may be invalid cache to destroy
392 TInt count = iCache.Count();
393 while(count > aIndexToInsert)
394 { //if there is cache where we wished to insert, it and everything above is now invalid
395 iCache[count-1].Close();
396 iCache.Remove(count-1);
400 return EFalse; //failed to insert
403 /** Will send a notification that a surface being unreferenced if it is not
404 found in any WindowElementSet
406 @pre Make sure the element being removed has already had its surfaceid set to NULL.
407 @param aSurfaceId The surface being removed
409 void CWindowElement::NotifyReleasingSurface(TSurfaceId aSurfaceId)
411 if (CWsTop::SearchDuplicateSurfaceId(aSurfaceId) == EFalse)
412 { //NOT found other instance of surface id, send notification now
413 TWservCrEvent crEvent(TWservCrEvent::ESurfaceUnreferenced, sizeof(aSurfaceId), &aSurfaceId);
414 TWindowServerEvent::NotifyDrawer(crEvent);
418 /** Called to place a element within the window.
419 It will either create a new element, recycle a element, or extend a element .
421 @pre UnassignPlacedElements has already been called.
422 @param aPlacedAttributes Returns the attributes of the surface placed.
423 @param aSurfaceConfiguration The surface configuration for the surface to place.
424 @param aUserDefinedRegion The user defined clipping of the window.
425 @param aScene Access to the scene.
426 @return Serious error, otherwise flags to describe what extra work needs doing.
428 @see CWindowElementSet::AssignPlacedElement
430 TInt CWindowElement::AssignPlacedElement(
431 TPlacedAttributes*& aPlacedAttributes,
432 const TSurfaceConfiguration& aSurfaceConfiguration,
433 const TRegion& aUserDefinedRegion,
438 if (iStartGcDrawingCount!=aGcDrawingCount)
440 FlagAssignedElementsDrawnOver(aGcDrawingCount);
443 if (iNonRedrawVisibleRegion.CheckError())
445 //CANNOT DO ANYTHING, LEAVE
446 return KErrNotReady; //error
450 aSurfaceConfiguration.GetExtent(surfaceExtent);
451 if (!aUserDefinedRegion.Intersects(surfaceExtent))
453 //clippedRegion.Close();
454 return KErrCancel; //no intersection, so nothing to do
457 TInt matchFound = -1;
459 //check 'current' (last placed element) to see if it is a match
460 if (iHighestReusedIndex != -1)
462 if (IsElementAMatch(*(iPlacedElements[iHighestReusedIndex].iElement),aSurfaceConfiguration,iPlacedElements[iHighestReusedIndex].Flags()) )
464 matchFound = iHighestReusedIndex;
468 if (matchFound == -1)
470 //check current unassigned elements to see if there is a match
471 for (TInt ii = (iHighestReusedIndex == -1)?0:iHighestReusedIndex+1; ii < iPlacedElements.Count(); ii++)
473 if (IsElementAMatch(*(iPlacedElements[ii].iElement),aSurfaceConfiguration,iPlacedElements[ii].Flags()))
480 TInt returnValue = 0;
481 if (matchFound == -1)
482 { //not found a element to extend
487 if (iPlacedElements.Count() == 0 || iHighestReusedIndex == -1 || matchFound == iHighestReusedIndex) //reached last re assigned element, so cannot reuse one
492 if (iPlacedElements[matchFound].Unassigned() && !iPlacedElements[matchFound].LockedConfig()) //found a element to reuse
499 if (matchFound != -1)
500 { //found a element to recycle
501 returnValue = AssignRecycleElement(matchFound,aSurfaceConfiguration,aUserDefinedRegion,aScene);
504 { //need to create a new element
505 returnValue = AssignCreateElement(aSurfaceConfiguration,aUserDefinedRegion,aScene);
509 { //found a candidate to extend
510 returnValue = AssignExtendElement(matchFound,aSurfaceConfiguration,aUserDefinedRegion,aScene);
512 if (returnValue < KErrNone)
513 { //failed with error
516 aPlacedAttributes = &iPlacedElements[iHighestReusedIndex];
517 iFlags|=returnValue; //update window wide flags
521 /** Recycle a element assigned to the window.
523 @param aIndex The index of the placed element to recycle.
524 @param aSurfaceConfiguration The surface configuration for the surface to place.
525 @param aUserDefinedRegion The user defined clipping of the window.
526 @param aScene Access to the scene.
527 @return Serious error, otherwise flags to describe what extra work needs doing.
529 @see CWindowElement::AssignPlacedElement
531 TInt CWindowElement::AssignRecycleElement(const TInt aIndex, const TSurfaceConfiguration& aSurfaceConfiguration, const TRegion& aUserDefinedRegion, MWsScene& aScene)
533 TInt returnValue = 0;
534 TInt error = KErrNone;
535 STACK_REGION clippedRegion;
537 aSurfaceConfiguration.GetExtent(surfaceExtent);
538 clippedRegion.AddRect(surfaceExtent);
539 clippedRegion.Intersect(aUserDefinedRegion);
540 if (clippedRegion.CheckError())
541 { //failed crucial calculation, leave while nothing has been changed!
542 clippedRegion.Close();
545 WS_ASSERT_DEBUG(!clippedRegion.IsEmpty(),EWsPanicRegion);
546 if (clippedRegion.IsEmpty())
551 //keep note of surface being removed
552 TSurfaceId oldSurfaceId = iPlacedElements[aIndex].iElement->ConnectedSurface();
554 //set element with new attributes
555 error = SetElement(*(iPlacedElements[aIndex].iElement),aSurfaceConfiguration,EFalse);
557 if (error < KErrNone)
558 { //must be a bad surface id or bad viewport, leave while nothing has been changed!
559 clippedRegion.Close();
563 returnValue|=EFastPath;
564 returnValue|=EPauseComposition;
565 //element has been altered, must make sure old surface unreferenced notify will be called
566 if (!iRemovedSurfacesValid)
567 { //didnt reserve space for surface ids, do search immediately
568 NotifyReleasingSurface(oldSurfaceId);
571 { //keep memory of surface id - search will be done during cleanup. Space already been reserved
572 iRemovedSurfaces.Append(oldSurfaceId);
575 //order placed element correctly in iPlacedElements
576 TPlacedAttributes tempAttributes = iPlacedElements[aIndex];
577 iPlacedElements.Remove(aIndex);
578 //keep cache up to date
579 if (iCache.Count() > aIndex)
581 iCache[aIndex].Close();
582 iCache.Remove(aIndex);
585 WS_ASSERT_DEBUG(iHighestReusedIndex != -1,EWsPanicRecycleElement);
586 if (iHighestReusedIndex == -1)
591 error = iPlacedElements.Insert(tempAttributes, iHighestReusedIndex);
592 if (error < KErrNone)
593 { //must destroy the element and leave. Old elements surface unregistration already dealt with
594 aScene.DestroySceneElement(tempAttributes.iElement);
595 clippedRegion.Close();
598 returnValue|=EResort; //element has been moved, will need a resort
599 //set placed element attributes
600 iPlacedElements[iHighestReusedIndex].SetUnassigned(EFalse);
601 iPlacedElements[iHighestReusedIndex].SetLockedConfig(ETrue);
602 iPlacedElements[iHighestReusedIndex].SetChangedClip(ETrue);
605 aSurfaceConfiguration.GetViewport(tempViewport);
607 TBool cacheSaved = EFalse;
608 if (iCache.Count() >= iHighestReusedIndex)
609 { //need to update cache with visible region of element
610 TBool insertSuccess = InsertCache(iHighestReusedIndex);
611 if (insertSuccess != EFalse)
613 iCache[iHighestReusedIndex].Copy(clippedRegion);
614 if (!iCache[iHighestReusedIndex].CheckError())
620 if (cacheSaved == EFalse)
621 { //need to set clipping to element immediately
622 TInt count = iCache.Count();
623 while(count > iHighestReusedIndex)
624 { //if there is cache where we wished to insert, it and everything above is now invalid
625 iCache[count-1].Close();
626 iCache.Remove(count-1);
630 clippedRegion.Close();
634 /** Create a new element for the placed surface.
636 @param aSurfaceConfiguration The surface configuration for the surface to place.
637 @param aUserDefinedRegion The user defined clipping of the window.
638 @param aScene Access to the scene.
639 @return Serious error, otherwise flags to describe what extra work needs doing.
641 @see CWindowElement::AssignPlacedElement
643 TInt CWindowElement::AssignCreateElement(const TSurfaceConfiguration& aSurfaceConfiguration, const TRegion& aUserDefinedRegion, MWsScene& aScene)
645 TInt returnValue = 0;
646 TInt error = KErrNone;
647 STACK_REGION clippedRegion;
649 aSurfaceConfiguration.GetExtent(surfaceExtent);
651 clippedRegion.AddRect(surfaceExtent);
652 clippedRegion.Intersect(aUserDefinedRegion);
653 if (clippedRegion.CheckError())
654 { //failed, return KErrNoMemory
655 clippedRegion.Close();
659 TPlacedAttributes newElement;
660 if (iHighestReusedIndex == -1)
662 error = iPlacedElements.Insert(newElement,0);
665 { //insert above current highest reused element
666 error = iPlacedElements.Insert(newElement,iHighestReusedIndex+1);
668 if (error >= KErrNone)
670 iHighestReusedIndex++;
674 //FAILED! LEAVE FUNCTION. nothing has been changed
675 clippedRegion.Close();
679 //create new element with new configuration
680 MWsElement* element = NULL;
681 TRAP(error,element = aScene.CreateSceneElementL());
682 if (error != KErrNone)
684 //FAILED! LEAVE FUNCTION. nothing has been changed
685 clippedRegion.Close();
689 element->SetGlobalAlpha( 0xFF );
691 iPlacedElements[iHighestReusedIndex].iElement = element;
692 error = SetElement(*element,aSurfaceConfiguration,EFalse);
693 if (error < KErrNone)
695 //must be a bad surface id or viewport. Leave now.
696 aScene.DestroySceneElement(element);
697 clippedRegion.Close();
698 iPlacedElements.Remove(iHighestReusedIndex);
699 iHighestReusedIndex--;
703 //Now certain that these must be flagged
704 returnValue|=EFastPath;
705 returnValue|=EPauseComposition;
706 returnValue|=EResort;
708 //set placed element attributes
709 iPlacedElements[iHighestReusedIndex].SetUnassigned(EFalse);
710 iPlacedElements[iHighestReusedIndex].SetChangedClip(ETrue);
711 iPlacedElements[iHighestReusedIndex].SetLockedConfig(ETrue);
714 aSurfaceConfiguration.GetViewport(tempViewport);
716 TBool cacheSaved = EFalse;
717 if (iCache.Count() >= iHighestReusedIndex)
718 { //should try to add new elements region to cache
719 TBool insertSuccess = InsertCache(iHighestReusedIndex);
720 if (insertSuccess != EFalse)
722 iCache[iHighestReusedIndex].Copy(clippedRegion);
723 if (!iCache[iHighestReusedIndex].CheckError())
729 if (cacheSaved == EFalse)
730 { //need to clear cache from this point onward, and set clipping to element immediately
731 TInt count = iCache.Count();
732 while(count > iHighestReusedIndex)
733 { //if there is cache where we wished to insert, it and everything above is now invalid
734 iCache[count-1].Close();
735 iCache.Remove(count-1);
739 //add the new element to the scene
740 error = aScene.InsertSceneElement(element, NULL); //place at the back. will get ordered correctly later
741 WS_ASSERT_DEBUG(error == KErrNone,EWsPanicSceneErrorIgnored);
742 clippedRegion.Close();
746 /** Extend an existing placed surface element.
748 @param The index of the element to extend.
749 @param aSurfaceConfiguration The surface configuration for the surface to place.
750 @param aUserDefinedRegion The user defined clipping of the window.
751 @param aScene Access to the scene.
752 @return Serious error, otherwise flags to describe what extra work needs doing.
754 @see CWindowElement::AssignPlacedElement
756 TInt CWindowElement::AssignExtendElement(const TInt aIndex, const TSurfaceConfiguration& aSurfaceConfiguration, const TRegion& aUserDefinedRegion, MWsScene& /*aScene*/)
758 TInt returnValue = 0;
759 TInt error = KErrNone;
760 STACK_REGION backupRegionAgainstFail;
761 STACK_REGION clippedRegion;
763 aSurfaceConfiguration.GetExtent(surfaceExtent);
764 TRect surfaceViewport;
765 aSurfaceConfiguration.GetViewport(surfaceViewport);
767 if (!iPlacedElements[aIndex].LockedConfig())
768 { //set element now. If viewport is invalid its ok to fail now
769 error = SetElement(*iPlacedElements[aIndex].iElement,aSurfaceConfiguration,ETrue);
770 if (error < KErrNone)
772 backupRegionAgainstFail.Close();
773 clippedRegion.Close();
778 if(iPlacedElements[aIndex].Unassigned() && !iPlacedElements[aIndex].Overlapping())
781 clippedRegion.Copy(aUserDefinedRegion);
785 clippedRegion.Copy(iNonRedrawVisibleRegion);
786 if (iCache.Count() > aIndex) //if cache is ok
788 clippedRegion.Intersect(iCache[aIndex]);
789 clippedRegion.Union(iCache[aIndex]); //previously added regions should be kept
793 STACK_REGION tempSurfaceBaseArea;
794 TRect destinationRect;
795 iPlacedElements[aIndex].iElement->GetDestinationRectangle(destinationRect);
796 tempSurfaceBaseArea.AddRect(destinationRect);
797 tempSurfaceBaseArea.Offset(surfaceExtent.iTl);
798 clippedRegion.Intersect(tempSurfaceBaseArea);
799 clippedRegion.Union(tempSurfaceBaseArea);
800 tempSurfaceBaseArea.Close();
802 backupRegionAgainstFail.Copy(clippedRegion);
803 //clipped region should contain something
804 clippedRegion.Union(aUserDefinedRegion);
806 clippedRegion.ClipRect(surfaceExtent);
808 if (backupRegionAgainstFail.CheckError() || clippedRegion.CheckError())
809 { //failed critical calculations, leave now before anything has been changed
810 clippedRegion.Close();
811 backupRegionAgainstFail.Close();
815 TBool setCache = EFalse;
816 if (iCache.Count() > aIndex) //if Cache is ok
820 compareReturn = TRegionExtend::Cast(iCache[aIndex]).TestDifference(clippedRegion);
822 if (compareReturn&TRegionExtend::EDiffers) //element has changed
824 iPlacedElements[aIndex].SetChangedClip(ETrue);
825 if (compareReturn&TRegionExtend::EAdd) //element has become revealed on the screen
827 returnValue|=EPauseComposition; //need to pause before the end of assign placed element
831 returnValue|=EFastPath; //fastpath and pause will be called during cleanup
835 //copy clipped region to cache
836 iCache[aIndex].Copy(clippedRegion); //can fail
838 if (iCache[aIndex].CheckError())
839 { //copy failed, remove cache from this element onwards
840 TInt count = iCache.Count();
841 while(count > aIndex)
842 { //if there is cache where we wished to insert, it and everything above is now invalid
843 iCache[count-1].Close();
844 iCache.Remove(count-1);
850 if (setCache == EFalse)
851 { //need to pause composition and update element immediately
852 returnValue|=EPauseComposition;
854 iPlacedElements[aIndex].SetUnassigned(EFalse);
855 iPlacedElements[aIndex].SetLockedConfig(ETrue);
856 iHighestReusedIndex = aIndex;
857 backupRegionAgainstFail.Close();
858 clippedRegion.Close();
862 /** Called at the end of a redraw to set cached changed to elements, and
863 removed elements that are still unassigned.
865 @param aScene Access to the scene.
866 @return Flags to describe what extra work needs doing.
867 @see CWindowElementSet::CleanUpPlacedElements
869 TInt CWindowElement::CleanUpPlacedElements(MWsScene& aScene, TInt aGcDrawingCount)
871 if (iStartGcDrawingCount!=aGcDrawingCount)
873 FlagAssignedElementsDrawnOver(aGcDrawingCount);
875 TInt returnFlags = iFlags;
877 { //copy flags to return, and ensure pause is flagged
878 returnFlags|=EPauseComposition;
881 TBool removeElements = EFalse;
882 //destroy placed element reference
883 for (ii=0; ii<iPlacedElements.Count(); ii++)
885 if (iPlacedElements[ii].Unassigned() && !iPlacedElements[ii].Overlapping())
887 returnFlags|=EFastPath; //need to invoke 1108
888 if (iRemovedSurfacesValid)
890 iRemovedSurfaces.Append(iPlacedElements[ii].iElement->ConnectedSurface());
891 iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
895 TSurfaceId tempId = iPlacedElements[ii].iElement->ConnectedSurface();
896 iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
897 NotifyReleasingSurface(tempId);
899 removeElements = ETrue;
904 { //there will be elements to remove
905 for (ii=0; ii<iPlacedElements.Count(); ii++)
907 if (iPlacedElements[ii].iElement->ConnectedSurface().IsNull())
909 aScene.DestroySceneElement(iPlacedElements[ii].iElement);
910 iPlacedElements.Remove(ii);
911 if (iCache.Count() > ii)
912 { //there is cache for this element, remove it
921 if (iRemovedSurfaces.Count() > 1)
922 { //remove duplicates of surface id from removal list
923 for (ii = 0; ii < iRemovedSurfaces.Count() - 1; ii++)
925 for (TInt jj = ii+1; jj < iRemovedSurfaces.Count(); jj++)
927 if (iRemovedSurfaces[ii] == iRemovedSurfaces[jj])
929 iRemovedSurfaces.Remove(jj);
936 //do global search for surface ids
937 while (iRemovedSurfaces.Count()>0)
939 TInt tempRemoval = iRemovedSurfaces.Count() - 1;
940 NotifyReleasingSurface(iRemovedSurfaces[tempRemoval]);
941 iRemovedSurfaces.Remove(tempRemoval);
944 iRemovedSurfaces.Reset();
946 //clip unassigned overlapping entries, mark as changedclip if region changed
947 for (ii=0; ii<iPlacedElements.Count(); ii++)
949 if (iPlacedElements[ii].Unassigned()) //non overlapping unassigned have already been destroyed
950 { //these must overlap
951 TBool failureOccured = EFalse;
952 if (iCache.Count() > ii)
954 //if cache region is entirely inside non redraw vis region, nothing to update!
956 TRegionExtend::Cast(iCache[ii]).TestDifference(iNonRedrawVisibleRegion);
957 if (changed&TRegionExtend::ESub)
958 { //the clipping will change
959 iCache[ii].Intersect(iNonRedrawVisibleRegion);
960 if (iCache[ii].CheckError())
962 failureOccured = ETrue;
968 iPlacedElements[ii].SetChangedClip(ETrue); //changed clipping
973 { //attempt getting element region to perform calculation and update element directly
974 STACK_REGION elementRegion;
975 TRect destinationRect;
976 iPlacedElements[ii].iElement->GetDestinationRectangle(destinationRect);
977 elementRegion.AddRect(destinationRect);
978 if (elementRegion.CheckError())
980 failureOccured = ETrue;
985 iPlacedElements[ii].iElement->GetDestinationRectangle(elementExtent);
986 elementRegion.Offset(elementExtent.iTl); //get element region into absolute coordinates
988 TRegionExtend::Cast(elementRegion).TestDifference(iNonRedrawVisibleRegion);
989 if (changed&TRegionExtend::ESub)
990 { //need to clip element back
991 elementRegion.Intersect(iNonRedrawVisibleRegion);
992 if (!elementRegion.CheckError())
994 elementRegion.Offset(-elementExtent.iTl); //put element region back into relative coords
995 returnFlags|=EFastPath;
999 failureOccured = ETrue;
1003 elementRegion.Close();
1006 { //need to release element
1007 TSurfaceId tempId = iPlacedElements[ii].iElement->ConnectedSurface();
1008 iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
1009 NotifyReleasingSurface(tempId);
1010 aScene.DestroySceneElement(iPlacedElements[ii].iElement);
1011 returnFlags|=EFailed;
1012 iPlacedElements.Remove(ii); //remove placed element entry, cache is already removed
1018 //update any elements marked as changedclip , clear all cache, clear flags
1019 for (ii=0; ii<iPlacedElements.Count(); ii++)
1021 if (iPlacedElements[ii].ChangedClip())
1023 returnFlags|=EFastPath; //need to invoke 1108
1026 if ( !(returnFlags&EFastPath)
1027 && iPlacedElements[ii].LockedConfig() && !iPlacedElements[ii].Unassigned()
1028 && iPlacedElements[ii].DrawnOver()!=iPlacedElements[ii].DrawnOverLast() )
1030 returnFlags|=EFastPath;
1032 iPlacedElements[ii].ClearRedrawFlags();
1033 if (iCache.Count() > ii)
1039 iNonRedrawVisibleRegion.Clear();
1040 iHighestReusedIndex = -1;
1044 /** Move all elements in this windowelement to above the supplied element in the scene
1046 @param aScene Access to the scene.
1047 @param aElement The element to set all this windows elements above.
1050 void CWindowElement::MoveToAboveGivenElement(MWsScene& aScene, MWsElement* aElement)
1052 MWsElement* newAboveElement = iBackgroundElement.iElement;
1053 MWsElement* previousAboveElement = aElement;
1054 if (newAboveElement)
1056 aScene.InsertSceneElement(newAboveElement, previousAboveElement); //background element is put on first
1057 previousAboveElement = newAboveElement;
1059 TInt placedElements = iPlacedElements.Count();
1060 if (placedElements != 0)
1062 for (TInt ii = 0; ii < placedElements; ii++)
1064 newAboveElement = iPlacedElements[ii].iElement;
1065 aScene.InsertSceneElement(newAboveElement, previousAboveElement); //place element above previous above element
1066 previousAboveElement = newAboveElement;
1071 /** Updates the elements extent, whether from a window movement or a change in window size
1073 @param aOffset The movement of the window. If NULL then the window has changed size.
1074 @see CWindowElementSet::UpdateElementExtent
1076 void CWindowElement::UpdateElementExtent(const TPoint* aOffset)
1078 if (aOffset) //window moved
1081 MWsElement* element = iBackgroundElement.iElement;
1082 TBool complete = EFalse;
1083 TInt placedElementDone = -1;
1086 if (!element && placedElementDone < iPlacedElements.Count()-1)
1088 placedElementDone++;
1089 element = iPlacedElements[placedElementDone].iElement;
1097 tempExtent = const_cast<CWsClientWindow&>(iWindow).GetOriginalDestElementRect();
1098 tempExtent.Move(*aOffset);
1099 element->SetDestinationRectangle(tempExtent);
1100 const_cast<CWsClientWindow&>(iWindow).SetOriginalDestElementRect(tempExtent);
1105 else //window changed size
1107 if (!iBackgroundElement.ExplicitExtent() && iBackgroundElement.iElement)
1109 iBackgroundElement.iElement->SetDestinationRectangle(iWindow.FullRect());
1110 const_cast<CWsClientWindow&>(iWindow).SetOriginalDestElementRect(iWindow.FullRect());
1115 /** Checks the windows placed elements when the windows visibility is changed.
1116 If the placed elements are no longer visible, they are removed.
1118 @param aRegion The new visible region of the window.
1119 @param aScene Access to the scene.
1120 @return ETrue if any elements have been removed, otherwise EFalse.
1121 @see CWindowElementSet::SetVisibleRegion
1123 TBool CWindowElement::SetVisibleRegion(const TRegion& aRegion, MWsScene& aScene)
1125 TBool retcode=EFalse;
1126 if (iPlacedElements.Count() == 0)
1128 return EFalse; //there is noting to do
1131 STACK_REGION tempRegion;
1132 for (TInt ii = 0; ii < iPlacedElements.Count(); ii++)
1136 iPlacedElements[ii].iElement->GetDestinationRectangle(tempExtent);
1137 tempRegion.AddRect(tempExtent);
1138 if (tempRegion.CheckError()) //if there was error getting region
1141 tempRegion.AddRect(tempExtent);
1144 { //offset basearea of element
1145 tempRegion.Offset(tempExtent.iTl);
1147 TInt regionReturn = TRegionExtend::Cast(aRegion).TestDifference(tempRegion);
1148 if (regionReturn&TRegionExtend::ENoIntersect)
1149 { //placed surface needs to be removed
1150 TSurfaceId tempId = iPlacedElements[ii].iElement->ConnectedSurface();
1151 iPlacedElements[ii].iElement->ConnectSurface(TSurfaceId::CreateNullId());
1152 NotifyReleasingSurface(tempId);
1153 aScene.DestroySceneElement(iPlacedElements[ii].iElement);
1154 retcode = ETrue; //a element has been removed
1155 iPlacedElements.Remove(ii);
1156 if (iCache.Count() > ii)
1157 { //keep cache up to date
1168 /** Checks if any of the windows element ids match the one in question.
1170 @param aSurfaceId The surface id to match against.
1171 @return ETrue if a match is found, otherwise EFalse.
1173 @see CWindowElementSet::SearchDuplicateSurfaceId
1175 TBool CWindowElement::SearchDuplicateSurfaceId(const TSurfaceId& aSurfaceId)
1177 if (iBackgroundElement.iElement)
1179 if (iBackgroundElement.iElement->ConnectedSurface() == aSurfaceId)
1184 if (iPlacedElements.Count() > 0)
1186 for (TInt ii = 0; ii < iPlacedElements.Count(); ii++)
1188 if (iPlacedElements[ii].iElement) //if removed without reserved space for ids, could be a null element
1190 if (iPlacedElements[ii].iElement->ConnectedSurface() == aSurfaceId)
1200 /** Sets the opacity for the background surface and if setting to 0 removes placed surfaces.
1202 @param aOpacity The opacity to set.
1203 @param aScene Access to the scene.
1204 @see CWindowElementSet::SetElementOpacity
1206 void CWindowElement::SetElementOpacity(TInt aOpacity, MWsScene& aScene)
1208 if (iBackgroundElement.iElement)
1211 iBackgroundElement.iElement->GetTargetRendererFlags(flags);
1212 flags |= MWsElement::EElementTransparencyGlobalAlpha;
1213 iBackgroundElement.iElement->SetTargetRendererFlags(MWsElement::EElementTransparencyGlobalAlpha);
1214 iBackgroundElement.iElement->SetGlobalAlpha(aOpacity);
1215 iBackgroundElement.SetConcealed(aOpacity==0);
1219 while (iPlacedElements.Count() > 0)
1220 { //remove any placed elements
1221 TInt placedElement = iPlacedElements.Count()-1;
1222 TSurfaceId tempId = iPlacedElements[placedElement].iElement->ConnectedSurface();
1223 iPlacedElements[placedElement].iElement->ConnectSurface(TSurfaceId::CreateNullId());
1224 NotifyReleasingSurface(tempId);
1225 aScene.DestroySceneElement(iPlacedElements[placedElement].iElement);
1226 iPlacedElements.Remove(placedElement);
1231 MWsElement* CWindowElement::Element() const
1233 return iBackgroundElement.iElement;
1237 // CWindowElementSet
1240 /** Destroys the set of window-element pairs.
1242 CWindowElementSet::~CWindowElementSet()
1244 ASSERT(iElements.Count() == 0);
1248 /** Basic NewL constructor
1249 @param aScene To allow access to the scene.
1250 @param aComposer To allow access to the composer.
1252 CWindowElementSet* CWindowElementSet::NewL(MWsScene& aScene)
1254 CWindowElementSet* wls = new (ELeave) CWindowElementSet(aScene);
1258 /** Sets a new background surface on the window supplied.
1259 Will remove any previous background surface.
1261 @param aWindow The window to place a background surface on.
1262 @return A reference to the new background surface attributes.
1264 TBackgroundAttributes& CWindowElementSet::AcquireBackgroundElementL(CWsClientWindow& aWindow)
1266 // Find the matching element.
1268 TInt result = FindEntry(aWindow, index);
1269 MWsElement* element;
1271 // If a background element is already associated with the window, then unregister
1272 // the surface. Create and add a new element to scene.
1273 // This will ensure that the element has default values.
1274 if (result != KErrNotFound && iElements[index]->iBackgroundElement.iElement)
1276 element = iElements[index]->iBackgroundElement.iElement;
1277 TSurfaceId surface = element->ConnectedSurface();
1278 element->ConnectSurface(TSurfaceId::CreateNullId());
1279 UnregisterSurface(surface);
1280 iScene.DestroySceneElement(element);
1287 // Allocate a new element and add it to the set and the scene
1288 element = iScene.CreateSceneElementL();
1291 if (result == KErrNotFound)
1293 CWindowElement* winelement = new CWindowElement(aWindow);
1296 iScene.DestroySceneElement(element);
1297 User::Leave(KErrNoMemory);
1299 returnCode = iElements.Insert(winelement, index);
1300 if(returnCode != KErrNone)
1303 iScene.DestroySceneElement(element);
1304 User::Leave(returnCode);
1308 // Use the element below the insertion point to decide where the element
1309 // goes in the scene
1310 returnCode = KErrNone;
1313 returnCode = iScene.InsertSceneElement(element, NULL);
1317 //Find highest element in window below
1319 TInt placedCount = iElements[index-1]->iPlacedElements.Count();
1320 if (placedCount > 0)
1322 below = iElements[index-1]->iPlacedElements[placedCount-1].iElement;
1325 { //below = above = background surface
1326 below = iElements[index-1]->iBackgroundElement.iElement;
1328 returnCode = iScene.InsertSceneElement(element, below);
1331 __ASSERT_DEBUG(returnCode == KErrNone, Panic(EWsPanicSceneErrorIgnored));
1333 iElements[index]->iBackgroundElement.iElement = element;
1334 return iElements[index]->iBackgroundElement;
1337 /** Removes the background element of the specified window.
1339 @param aWindow The window to remove the background element from.
1340 @param aWindowClosing ETrue if aWindow is in between closing state.
1341 @return KErrNone on success or a system-wide error code.
1343 TInt CWindowElementSet::ReleaseBackgroundElement(const CWsClientWindow& aWindow, TBool aWindowClosing /*= EFalse*/)
1347 TInt err = FindEntry(aWindow, index, aWindowClosing);
1348 __ASSERT_DEBUG(err != KErrNotFound, Panic(EWsPanicNoWindowElement));
1352 CWindowElement* winElement=iElements[index];
1353 if (winElement->iPlacedElements.Count() == 0)
1354 { //destroy the window entry
1355 DestroyWindowElementEntry(index);
1358 { // just destroy the background
1359 if (winElement->iBackgroundElement.iElement)
1361 winElement->iBackgroundElement.Destroy(iScene, ETrue);
1368 /** Destroys the element associated with this set of attributes
1370 @param aScene To allow access to the scene.
1371 @param aUnregister Whether to call unregister on the surface id.
1372 @return ETrue if a element existed to be destroyed, otherwise EFalse.
1374 TBool TAttributes::Destroy(MWsScene& aScene, TBool aUnregister)
1376 MWsElement* element = iElement;
1379 TSurfaceId surface = element->ConnectedSurface();
1380 element->ConnectSurface(TSurfaceId::CreateNullId());
1383 aScene.UnregisterSurface(surface);
1385 aScene.DestroySceneElement(element);
1388 return element!=NULL;
1391 /** Destroys all elements associated with the window and removes the window from the element set
1393 @param aWinElementIndex The index of the window in the element set
1395 void CWindowElementSet::DestroyWindowElementEntry(const TInt aWinElementIndex)
1397 CWindowElement* winElement=iElements[aWinElementIndex];
1398 const CWsClientWindow& window = winElement->iWindow;
1399 winElement->iBackgroundElement.Destroy(iScene, ETrue);
1400 for (TInt placedIndex=0,maxindex=winElement->iPlacedElements.Count();placedIndex<maxindex;placedIndex++)
1402 winElement->iPlacedElements[placedIndex].Destroy(iScene, EFalse);
1404 winElement->iPlacedElements.Close();
1406 iElements.Remove(aWinElementIndex);
1407 window.Redraw()->SetHasElement(EFalse);
1408 window.Screen()->ElementRemoved();
1411 /** Removes all elements in the associated window.
1413 @param aWindow The window to remove the elements for.
1414 @param aWindowClosing ETrue if aWindow is in between closing state.
1415 @return KErrNone on success, KErrNotFound if there was no windowelement entry.
1417 TInt CWindowElementSet::ReleaseAllElements(const CWsClientWindow& aWindow, TBool aWindowClosing /*= EFalse*/)
1421 TInt err = FindEntry(aWindow, index, aWindowClosing);
1422 __ASSERT_DEBUG(err != KErrNotFound, Panic(EWsPanicNoWindowElement));
1425 DestroyWindowElementEntry(index);
1430 /** For every window element, checks if any of the windows element ids match the one in question.
1432 @param aSurfaceId The surface id to match against.
1433 @return ETrue if a match is found, otherwise EFalse.
1435 @see CWindowElement::SearchDuplicateSurfaceId
1437 TBool CWindowElementSet::SearchDuplicateSurfaceId(const TSurfaceId& aSurfaceId)
1439 TInt windowElements = iElements.Count();
1440 for (TInt ii = 0; ii < windowElements; ii++)
1442 if (iElements[ii]->SearchDuplicateSurfaceId(aSurfaceId))
1450 /** For a window, sets the opacity for the background surface and if setting to
1451 0 removes placed surfaces.
1453 @param aWindow The window to perform the function on.
1454 @param aOpacity The opacity to set.
1455 @see CWindowElement::SetElementOpacity
1457 void CWindowElementSet::SetElementOpacity(CWsClientWindow& aWindow, TInt aOpacity)
1459 WS_ASSERT_DEBUG(aOpacity == 0 || aOpacity == 255,EWsPanicUnexpectedOpacity);
1460 return FindElement(aWindow)->SetElementOpacity(aOpacity, iScene);
1463 /** Finds all elements within the window that overlap with the region and
1464 marks them as unassigned.
1465 If any part of these unassigned elements are outside the region, they are also
1466 marked overlapping and locked config.
1468 @param aRedrawRegion The region in which elements will be unassigned
1469 @param aWindow The window to perform the function on.
1470 @return KErrNone on success or KErrNoMemory if critical calculations could not be made.
1471 @see CWindowElement::UnassignPlacedElements
1473 TInt CWindowElementSet::UnassignPlacedElements(const TRegion& aRedrawRegion, const CWsClientWindow& aWindow, TInt aGcDrawingCount)
1475 return FindElement(aWindow)->UnassignPlacedElements(aRedrawRegion, aGcDrawingCount);
1478 /** Called to place a element within the window.
1479 It will either create a new element, recycle a element, or extend a element .
1481 @param aPlacedAttributes Returns the attributes of the surface placed.
1482 @param aSurfaceConfiguration The surface configuration for the surface to place.
1483 @param aUserDefinedRegion The user defined clipping of the window.
1484 @param aWindow The window to perform the function on.
1485 @return KErrNone on success or a system-wide error code.
1486 @see CWindowElement::AssignPlacedElement
1488 TInt CWindowElementSet::AssignPlacedElement(
1489 TPlacedAttributes*& aPlacedAttributes,
1490 const TSurfaceConfiguration& aSurfaceConfiguration,
1491 const TRegion& aUserDefinedRegion,
1492 const CWsClientWindow& aWindow,
1493 TInt aGcDrawingCount )
1496 TInt error = KErrNone;
1497 TInt result = FindEntry(aWindow, index);
1498 TBool insertedElement = EFalse;
1499 if (result == KErrNotFound)
1501 CWindowElement* winelement = new CWindowElement(aWindow);
1504 return KErrNoMemory; //memory error
1506 error = iElements.Insert(winelement, index);
1507 if (error == KErrNone)
1509 insertedElement = ETrue;
1510 aWindow.Redraw()->SetHasElement(ETrue);
1511 STACK_REGION windowRegion = aWindow.Abs();
1512 if (!windowRegion.CheckError())
1514 error = iElements[index]->UnassignPlacedElements(windowRegion, aGcDrawingCount);
1518 error = KErrNoMemory;
1520 windowRegion.Close();
1527 if (error == KErrNone)
1530 TInt assignReturn = iElements[index]->AssignPlacedElement(aPlacedAttributes, aSurfaceConfiguration,
1531 aUserDefinedRegion, iScene, aGcDrawingCount);
1532 error = assignReturn; //return assign flags
1534 if (error < KErrNone && insertedElement)
1535 { //remove this element that has just been created
1536 aWindow.Redraw()->SetHasElement(EFalse);
1537 iElements.Remove(index);
1542 /** Marks all elements that have been assigned since the unassign as drawn over.
1544 @param aWindow The window to perform the function on.
1545 @see CWindowElement::FlagAssignedElementsDrawnOver
1547 //void CWindowElementSet::FlagAssignedElementsDrawnOver(const CWsClientWindow& aWindow)
1549 // FindElement(aWindow)->FlagAssignedElementsDrawnOver();
1552 /** Called at the end of a redraw to set cached changed to elements, and
1553 removed elements that are still unassigned.
1555 @param aWindow The window to perform the function on.
1556 @return Flags to describe what extra work needed doing.
1557 @see CWindowElement::CleanUpPlacedElements
1559 TInt CWindowElementSet::CleanUpPlacedElements(const CWsClientWindow& aWindow, TInt aGcDrawingCount)
1562 TInt error = FindEntry(aWindow,index);
1565 WS_ASSERT_DEBUG(0,EWsPanicNoWindowElement);
1568 TInt returnFlags = iElements[index]->CleanUpPlacedElements(iScene, aGcDrawingCount);
1570 if (!iElements[index]->iBackgroundElement.iElement && iElements[index]->iPlacedElements.Count() == 0)
1571 { //remove window entry
1572 DestroyWindowElementEntry(index);
1576 if (returnFlags&CWindowElement::EResort)
1577 { //need to sort elements
1578 //find front most visible element in window behind, invoke sort by z order
1580 { //this element is backmost window
1581 iElements[index]->MoveToAboveGivenElement(iScene,NULL);
1584 { //place infront of highest element behind
1585 if (iElements[index-1]->iPlacedElements.Count() == 0)
1586 { //top element of element behind must be the background element
1587 iElements[index]->MoveToAboveGivenElement(iScene,iElements[index-1]->iBackgroundElement.iElement);
1590 { //top element of element behind must be highest placed element
1591 TInt placedCount = iElements[index-1]->iPlacedElements.Count();
1592 iElements[index]->MoveToAboveGivenElement(iScene,iElements[index-1]->iPlacedElements[placedCount-1].iElement);
1597 if (returnFlags&CWindowElement::EFailed)
1598 { //visible elements may have been deleted, error to suggest a full window redraw
1604 /** Counts how many elements there are in the set.
1606 @return The number of elements in the set.
1608 TInt CWindowElementSet::Count() const
1610 return iElements.Count();
1613 /** Returns the background attributes for the specified window
1615 @param aWindow The window to perform the function on.
1616 @return The background surface attributes. If the window has no elementset entry, returns NULL.
1618 TBackgroundAttributes* CWindowElementSet::FindBackgroundElement(const CWsClientWindow& aWindow)
1621 TInt err = FindEntry(aWindow, index);
1623 if (err != KErrNotFound)
1625 return &(iElements[index]->iBackgroundElement);
1630 /** Returns the contents of the element data associated with the input window.
1631 If this method is successful, then neither pointer will be NULL.
1633 @return standard symbian error code.
1634 @param aWindow window to find
1635 @param aBackAttr backgroud surface attributes associated with the window
1636 @param aPlacedAttr array of placed surface attributes associated with the window.
1638 TInt CWindowElementSet::FindElements( CWsClientWindow const &aWindow,
1639 TBackgroundAttributes const * & aBackAttr,
1640 RArray<class TPlacedAttributes> const * & aPlacedAttr )
1644 TInt err = FindEntry(aWindow, index);
1645 // __ASSERT_DEBUG(err != KErrNotFound, Panic(EWsPanicNoWindowElement));
1647 // return *iElements[index].iElement;
1648 if (err >= KErrNone)
1650 aBackAttr=&iElements[index]->iBackgroundElement;
1651 aPlacedAttr=&iElements[index]->iPlacedElements;
1662 /** Registers the surface with the compositor.
1663 @param aSurface The surface id to register.
1664 @return KErrNone if successful, KErrNoMemory if registration fails due to low
1665 memory, KErrNotSupported if the surface is not compatible with
1666 this compositor or KErrBadHandle if the given surface ID does not
1667 represent a valid surface. KErrArgument is returned if the
1668 surface does not have both dimensions less than 32 767 pixels.
1670 TInt CWindowElementSet::RegisterSurface(const TSurfaceId& aSurface)
1673 return iScene.RegisterSurface(aSurface);
1676 /** Unregisters the surface with the compositor.
1678 @param aSurface The surface id to register.
1679 @return KErrNone if successful. KErrInUse if the surface is
1680 used by a layer or layers. KErrBadHandle if the surface
1681 is not currently registered. KErrArgument if
1682 the surface ID is a NULL ID.
1685 void CWindowElementSet::UnregisterSurface(const TSurfaceId& aSurface)
1688 TInt returnCode = iScene.UnregisterSurface(aSurface);
1689 __ASSERT_DEBUG((returnCode==KErrNone || returnCode==KErrInUse || returnCode==KErrBadHandle), Panic(EWsPanicSceneErrorIgnored));
1692 /** Sorts the array elements into the same order as the windows are in the
1693 hierarchy. Use after window hierarchy has been modified, to update the scene
1695 @return EFalse if element order unchanged, ETrue if order may have changed.
1697 TBool CWindowElementSet::SortByZOrder()
1699 if (iElements.Count() < 2)
1701 // Early out for the very common cases where there are zero or one
1702 // elements, which cannot therefore be out of order.
1706 // The approach being used is to first just sort the array, then update the
1707 // scene order afterwards. This is simple to code and since there are not
1708 // expected to be many elements, it should be perfectly good enough.
1710 TLinearOrder<CWindowElement> order(WindowOrder);
1711 iElements.Sort(order);
1713 TBool orderChanged = EFalse;
1715 TInt elementCount = iElements.Count();
1716 for (TInt index = 0; index < elementCount; index++)
1724 //Find highest element in window below
1725 TInt placedCount = iElements[index-1]->iPlacedElements.Count();
1726 if (placedCount > 0)
1728 below = iElements[index-1]->iPlacedElements[placedCount-1].iElement;
1731 { //below = above = background surface
1732 below = iElements[index-1]->iBackgroundElement.iElement;
1735 //Find lowest element in above window element
1737 if (iElements[index]->iBackgroundElement.iElement != NULL)
1738 { //use background element
1739 above = iElements[index]->iBackgroundElement.iElement; //background element will be bottom
1741 else //use bottom element of placed surfaces
1743 //above = iElements[index]->iPlacedElements[iElements[index]->iPlacedElements.Count()-1].iElement; //first or last placed element is bottom?
1744 above = iElements[index]->iPlacedElements[0].iElement;
1746 if (above->ElementBelow() != below)
1748 //CALL below window element function to move all elements above 'below'
1749 iElements[index]->MoveToAboveGivenElement(iScene, below);
1750 orderChanged = ETrue;
1753 return orderChanged;
1756 /** Processes the specified windows placed elements, for when windows visibility is changed.
1757 If the placed elements are no longer visible, they are removed.
1759 @param aWindow The window to call the function on.
1760 @return Positive if any elements have been removed, zero if not, or errorcode.
1761 @see CWindowElement::SetVisibleRegion
1763 TInt CWindowElementSet::SetVisibleRegion(CWsClientWindow& aWindow)
1766 TInt find = FindEntry(aWindow,index);
1767 WS_ASSERT_DEBUG(find>=KErrNone,EWsPanicNoWindowElement);
1768 TBool ret = iElements[index]->SetVisibleRegion(aWindow.VisibleRegion(), iScene);
1770 if (!iElements[index]->iBackgroundElement.iElement && iElements[index]->iPlacedElements.Count() == 0)
1772 DestroyWindowElementEntry(index);
1777 /** Updates the specified windows elements extent, either from a window movement or a change in window size
1779 @param aWindow The window to call the function on.
1780 @param aOffset The movement of the window. If NULL then the window has changed size.
1781 @see CWindowElement::UpdateElementExtent
1783 void CWindowElementSet::UpdateElementExtent(const CWsClientWindow& aWindow, const TPoint* aOffset)
1785 FindElement(aWindow)->UpdateElementExtent(aOffset);
1788 /** Method to fill in a TSurfaceConfiguration from a scene element.
1790 @param aConfiguration Surface configuration to fill in.
1791 @param aElement Element to get information from.
1792 @return Once multiple versions of TSurfaceConfiguration are available, KErrNotSupported if configuration supplied is too small.
1794 TInt CWindowElementSet::GetConfiguration(TSurfaceConfiguration& aConfiguration,MWsElement& aElement)
1796 TSurfaceId tempSurfaceId = aElement.ConnectedSurface();
1797 aConfiguration.SetSurfaceId(tempSurfaceId);
1799 //Convert and copy orientation
1800 aConfiguration.SetOrientation(ElementToGcRotation(aElement.SourceRotation()));
1802 //Convert and copy flip
1803 TBool flip = aElement.SourceFlipping();
1804 aConfiguration.SetFlip(flip);
1806 //Convert and copy viewport
1808 aElement.GetSourceRectangle(tempViewport);
1809 aConfiguration.SetViewport(tempViewport);
1811 aElement.GetDestinationRectangle(tempExtent);
1812 aConfiguration.SetExtent(tempExtent);
1813 return KErrNone; //Could fail if there are multiple versions of TSurfaceConfiguration
1816 /** Returns the window element entry for the specified window.
1818 @param aWindow The window to call the function on.
1820 CWindowElement* CWindowElementSet::FindElement(const CWsClientWindow& aWindow) const
1823 TInt error = FindEntry(aWindow,index);
1824 if (error == KErrNone)
1825 return iElements[index];
1826 WS_ASSERT_DEBUG(EFalse,EWsPanicNoWindowElement);
1830 /** Creates a set of window-element pairs for this scene.
1832 @param aScene To allow access to the scene.
1833 @param aComposer To allow access to the composer.
1835 CWindowElementSet::CWindowElementSet(MWsScene& aScene) :
1839 /** Searches for the entry in iElements with the given window
1841 @param aWindow The window to find the entry of / where it should be inserted.
1842 @param aIndex aIndex is set to entry found or the insertion point, respectively.
1843 @param aLinearSearch ETrue if a linear search of the window element set is required.
1844 @return KErrNone if found or KErrNotFound.
1846 TInt CWindowElementSet::FindEntry(const CWsClientWindow& aWindow, TInt& aIndex, TBool aLinearSearch /*= EFalse*/) const
1848 CWindowElement winelement(aWindow);
1850 // The array order makes use of the parent pointer, which gets reset during
1851 // window shutdown, so if it is clear fall back to a linear search.
1852 if (!aWindow.BaseParent() || aLinearSearch)
1854 TIdentityRelation<CWindowElement> match(WindowMatch);
1856 aIndex = iElements.Find(&winelement, match);
1857 return (aIndex == KErrNotFound) ? KErrNotFound : KErrNone;
1861 TLinearOrder<CWindowElement> order(WindowOrder);
1862 return iElements.FindInOrder(&winelement, aIndex, order);
1866 /** Used to find an entry in the set when order cannot be used.
1868 @param aFirst First windowelement to compare.
1869 @param aSecond Second windowelement to compare.
1870 @return ETrue if the entries are the same, and EFalse if they are not.
1872 TBool CWindowElementSet::WindowMatch(const CWindowElement& aFirst, const CWindowElement& aSecond)
1874 return (&aFirst.iWindow == &aSecond.iWindow);
1878 /** Used to determine the order of entries in the set.
1880 @param aFirst First windowelement to compare.
1881 @param aSecond Second windowelement to compare.
1882 @return zero if the entries are the same, a negative value if
1883 aFirst is behind aSecond or a positive value if aFirst is in front of aSecond.
1885 TInt CWindowElementSet::WindowOrder(const CWindowElement& aFirst, const CWindowElement& aSecond)
1889 if (&aFirst.iWindow != &aSecond.iWindow)
1891 result = aFirst.iWindow.IsInfrontOf(&aSecond.iWindow) ? 1 : -1;
1898 /** Re-sends the extents for all the elements (in this window) to the scene
1899 * This allows the renderstage to re-scale those element extents
1903 void CWindowElement::ResubmitAllElementExtents()
1905 if (MWsElement* element=iBackgroundElement.iElement)
1907 TRect extent(TRect::EUninitialized);
1908 element->GetDestinationRectangle(extent);
1909 element->SetDestinationRectangle(extent);
1911 TInt elementCount = iPlacedElements.Count();
1912 for (TInt index = 0; index < elementCount; index++)
1914 if (MWsElement* element=iPlacedElements[index].iElement)
1916 TRect extent(TRect::EUninitialized);
1917 element->GetDestinationRectangle(extent);
1918 element->SetDestinationRectangle(extent);
1923 /** Re-sends the extents for all the elements (in all the windows) to the scene
1924 * This allows the renderstage to re-scale those element extents
1928 void CWindowElementSet::ResubmitAllElementExtents()
1930 TInt elementCount = iElements.Count();
1931 for (TInt index = 0; index < elementCount; index++)
1933 iElements[index]->ResubmitAllElementExtents();
1941 /** Returns background attributes for the specified window index.
1942 For use with debug client interface.
1944 @param aWin Window index to get the surface attributes from.
1945 @return Background attributes for the specified window. NULL if the window index is invalid.
1947 const TBackgroundAttributes* CWindowElementSet::DebugBackgroundAt(TUint aWin)const
1951 return &iElements[aWin]->iBackgroundElement;
1954 /** Returns the client window for the specified window index.
1955 For use with debug client interface.
1957 @param aWin Window index to get the client window from.
1958 @return Client window for the specified window. NULL if the window index is invalid.
1960 const CWsClientWindow* CWindowElementSet::DebugWindowAt(TUint aWin)const
1964 return &iElements[aWin]->iWindow;
1967 /** Returns how many placed surfaces are on the specified window
1968 For use with debug client interface.
1970 @param aWin Window index to get the client window from.
1971 @return Amount of placed surfaces for the specified window index
1973 TInt CWindowElementSet::DebugPlacedCountAt(TUint aWin)const
1977 return iElements[aWin]->iPlacedElements.Count();
1980 /** Returns the placed attributes for the specified placed surface index in the
1981 specified window index.
1982 For use with debug client interface.
1984 @param aWin Window index to get the client window from.
1985 @param aPlace Placed surface index to get the placed surface attributes
1986 @return Placed surface attributes. NULL if either index was invalid.
1988 const TPlacedAttributes* CWindowElementSet::DebugPlacedAt(TUint aWin,TUint aPlace)const
1992 if (aPlace>=iElements[aWin]->iPlacedElements.Count())
1994 return &iElements[aWin]->iPlacedElements[aPlace];
1997 MWsElement* CWindowElementSet::GetElementFromWindow(const CWsClientWindow& aWindow) const
1999 CWindowElement* windowElement = FindElement(aWindow);
2001 return windowElement->Element();