os/graphics/windowing/windowserver/nga/SERVER/openwfc/wnredraw.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 1995-2010 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Window redraw code, three sorts of redrawing are supported
    15 // Sending a redraw message to the client (see redrawmsgwindow.cpp)
    16 // Drawing from backup bitmap
    17 // Simply clearing the window
    18 // 
    19 //
    20 
    21 #include "wnredraw.h"
    22 #include "server.h"
    23 #include "playbackgc.h"
    24 #include "wstop.h"
    25 #include "ANIM.H"
    26 #include "EVQUEUE.H"
    27 #include <s32mem.h>
    28 #include <gdi.h>
    29 #include "panics.h"
    30 #include "inifile.h"
    31 #include "rootwin.h"
    32 #include "EVENT.H"
    33 #include "wstypes.h"
    34 #include <graphics/surface.h>
    35 #include <graphics/wselement.h>
    36 #include <graphics/wsscreendevice.h>
    37 #include "windowelementset.h"
    38 
    39 struct TFadingParams
    40 	{
    41 	TUint8 blackMap;
    42 	TUint8 whiteMap;
    43 	};
    44 
    45 CWsWindowRedraw::CWsWindowRedraw(CWsWindow *aWin) : iWsWin(aWin)
    46 	{
    47 	}
    48 
    49 CWsWindowRedraw::~CWsWindowRedraw()
    50 	{
    51 	if (iWsWin->WsOwner())
    52 		{
    53 		iWsWin->WsOwner()->RedrawQueue()->RemoveInvalid(this);
    54 		}
    55 	if (HasElement())
    56 		{
    57 		iWsWin->Screen()->WindowElements().ReleaseAllElements(*CliWin());
    58 		}
    59 	}
    60 
    61 void CWsWindowRedraw::ConstructL()
    62 	{
    63 	}
    64 
    65 const TRegion& CWsWindowRedraw::InvalidArea() const
    66 	{
    67 	return(nullRegion);
    68 	}
    69 
    70 const TRegion &CWsWindowRedraw::BaseDrawRegion() const
    71 	{
    72 	return(iWsWin->VisibleRegion());
    73 	}
    74 
    75 void CWsWindowRedraw::ClipInvalidRegion(const TRect &)
    76 	{
    77 	}
    78 
    79 void CWsWindowRedraw::Resize(const TSize &, const TSize &)
    80 	{
    81 	}
    82 
    83 void CWsWindowRedraw::SetReply(TInt aReply)
    84 	{
    85 	iWsWin->WsOwner()->SetReply(aReply);
    86 	}
    87 
    88 void CWsWindowRedraw::OwnerPanic(TClientPanic aPanic)
    89 	{
    90 	iWsWin->OwnerPanic(aPanic);
    91 	}
    92 
    93 CWsBackedUpWindow *CWsWindowRedraw::Backup() const
    94 	{
    95 	return(NULL);
    96 	}
    97 
    98 void CWsWindowRedraw::Scroll(const TRect &, const TPoint &,const TRect &)
    99 	{
   100 	}
   101 
   102 void CWsWindowRedraw::UpdateAnimArea()
   103 	{
   104 	}
   105 
   106 void CWsWindowRedraw::PrepareForResizeL(const TSize& /* aNewSize */, TSize& /* aOldSize */)
   107 	{
   108 	}
   109 
   110 TBool CWsWindowRedraw::DrawCommand(CWsGc*,const TAny*)
   111 	{
   112 	return ETrue;
   113 	}
   114 
   115 void CWsWindowRedraw::GcAttributeChange(CWsGc*,const TAny*)
   116 	{
   117 	}
   118 
   119 void CWsWindowRedraw::GcDeactivate(CWsGc*)
   120 	{
   121 	}
   122 
   123 CFbsDevice* CWsWindowRedraw::OutputDevice() const
   124 	{
   125 	return NULL;
   126 	}
   127 
   128 void CWsWindowRedraw::ClientExposing()
   129 	{
   130 	}
   131 
   132 void CWsWindowRedraw::ClearRedrawStore(TBool)
   133 	{}
   134 
   135 void CWsWindowRedraw::PreDrawWindow(MWsGraphicsContext* aGc, const TRegion& aWindowRegion)
   136 	{
   137 	WS_ASSERT_DEBUG(iRedrawRegion == NULL, EWsPanicScheduledRedraw);
   138 	iRedrawRegion = &aWindowRegion;
   139 	CPlaybackGc::Instance()->SetTargetRegion(iRedrawRegion);
   140 	CWsClient::iCurrentCommand.iOpcode=0;
   141 	CPlaybackGc::Instance()->Activate(CliWin(), aGc, iRedrawRegion);
   142 	}
   143 
   144 void CWsWindowRedraw::PostDrawWindow(MWsGraphicsContext* aGc, const TRegion& aWindowChildNodeRegion)
   145 	{
   146 	WS_ASSERT_DEBUG(iRedrawRegion, EWsPanicScheduledRedraw);
   147 	CPlaybackGc::Instance()->Deactivate();
   148 	CPlaybackGc::Instance()->SetTargetRegion(NULL);
   149 	
   150 	if(!Screen()->ChangeTracking())
   151 		{
   152 		DoFade(*iRedrawRegion);
   153 		}
   154 
   155 	AnnotateWindowRedrawEnd(*iWsWin);	
   156 	
   157 	DrawWindowAnims(aGc, aWindowChildNodeRegion);
   158 	DrawCursorAndSprites(aGc, aWindowChildNodeRegion);
   159 	iRedrawRegion = 0;
   160 	}
   161 
   162 void CWsWindowRedraw::Fade(MWsGraphicsContext * aGc, const TRegion& aRegion)
   163 	{
   164 	LOG_WINDOW_FADE_START(WsWin());	
   165 	AnnotateWindowRedrawStart(*iWsWin, aRegion);
   166 	
   167 	aGc->Reset();
   168 	DoFade(aRegion);
   169 	
   170 	AnnotateWindowRedrawEnd(*iWsWin);		
   171 	LOG_WINDOW_FADE_END(WsWin());	
   172 	}
   173 
   174 void CWsWindowRedraw::DoFade(const TRegion& aRegion)
   175 	{
   176 	if( CWsTop::IsFadeEnabled() && iWsWin && iWsWin->FadeCount()>0 && !(iWsWin->IsNonFading()) && !(iWsWin->FadableRegion().IsEmpty()) && !(iWsWin->IsDSAHost()) )
   177 		{
   178 		MWsFader* fader = static_cast<MWsFader*>(iWsWin->Screen()->ResolveObjectInterface(KMWsFader));
   179 		if(fader)
   180 			{
   181 		  	TFadingParams parameters;
   182 	  		iWsWin->GetFadingParams(parameters.blackMap,parameters.whiteMap);
   183 	      	TPckgBuf<TFadingParams> buf(parameters);
   184 	      	fader->SetFadingParameters(buf);
   185 		  	// Only fade the region that hasn't been faded before
   186 	  		STACK_REGION fdRgn;
   187 	  		fdRgn.Copy( aRegion );
   188 			fdRgn.Intersect( iWsWin->FadableRegion() );
   189 			if(!fdRgn.CheckError())
   190 				{
   191 				fader->FadeArea( fdRgn );
   192 				LOG_WINDOW_FADE_REGION(&fdRgn);
   193 				}
   194 	  		fdRgn.Close();
   195 	      	}
   196 		}
   197 	}
   198 
   199 void CWsWindowRedraw::DrawWindowAnims(MWsGraphicsContext * aGc, const TRegion& aRegion)
   200 	{
   201 	if (iWsWin->iAnimList)
   202 		{
   203 		// If an anim panics, it will leave and set the panic flag on the client
   204 		// The client itself won't actually panic yet, and we don't want to leave from here.
   205 		TRAP_IGNORE(DrawWindowAnimsL(aGc, aRegion));
   206 		}
   207 	}
   208 
   209 void CWsWindowRedraw::DrawWindowAnimsL(MWsGraphicsContext * aGc, const TRegion& aRegion)
   210 	{
   211 	for (CWsAnim * anim = iWsWin->iAnimList; anim; anim = anim->Next())
   212 		{
   213 		AnnotateWindowAnimRedrawStart(*iWsWin, *anim, aRegion);
   214 		
   215 		//Animate and redraw
   216 		TRAPD(err,anim->RedrawWindowAnimL(Screen()->Now(), aGc, &aRegion));
   217 		if(err!=KErrNone)
   218 			{
   219 			AnnotateWindowAnimRedrawEnd(*iWsWin, *anim);
   220 			anim->Panic(EWservPanicAnimLeave);
   221 			return;
   222 			}
   223 		
   224 		AnnotateWindowAnimRedrawEnd(*iWsWin, *anim);
   225 		}
   226 	}
   227 
   228 void CWsWindowRedraw::DrawCursorAndSprites(MWsGraphicsContext * aGc, const TRegion& aRegion)
   229 	{	
   230 	// Draw standard text cursor if required
   231 	RWsTextCursor* const cursor = CWsTop::CurrentTextCursor();
   232 	if (!iWsWin->Screen()->ChangeTracking() && cursor && cursor->Win() == iWsWin && cursor->IsStandardCursorActive())
   233 		{
   234 		// Standard text cursor is active on this window
   235 		const TBool flashing = cursor->IsFlashing();
   236 		TFlashState flashState = EFlashOn;
   237 		if (flashing)
   238 			{
   239 			flashState = cursor->CurrentCursorFlashState();
   240 			}
   241 		if (flashState == EFlashOn)
   242 			{
   243 			// Cursor should be visible, so draw it
   244 			cursor->Draw(aRegion);
   245 			}
   246 		if (flashing)
   247 			{
   248 			// Reschedule to flash the standard cursor on or off
   249 			Screen()->ScheduleAnimation(ETextCursor, cursor->RectRelativeToScreen(), Screen()->SpriteManager()->NextCursorFlashStateChange(), 0, 0, iWsWin);
   250 			}
   251 		}
   252 
   253 	for (CWsSpriteBase * sprite = iWsWin->iSpriteList; sprite; sprite = sprite->Next())
   254 		{
   255         TBool hasRedrawBegun = EFalse;
   256 		STACK_REGION redrawRegion;
   257 		sprite->CalcRedrawRegion(aRegion, redrawRegion);
   258 		if(redrawRegion.CheckError() || !redrawRegion.IsEmpty())
   259 			{
   260 			if (sprite->IsFlashingEnabled() || sprite->IsDirty() || sprite->HasAnimation())
   261 				{
   262                 if (sprite->IsDirty() || sprite->HasAnimation())
   263                     {
   264                     AnnotateSpriteRedrawStart(*iWsWin, *sprite, redrawRegion);
   265                     hasRedrawBegun = ETrue;
   266                     }
   267 				
   268 				if(sprite->HasAnimation())
   269 					{
   270 					CWsAnim* anim = static_cast<CWsSprite*>(sprite)->iAnim;
   271 					WS_ASSERT_DEBUG(anim,EWsPanicAnim);
   272 					
   273 					//Animate and...
   274 					TRAPD(err, anim->AnimateSpriteAnimL(Screen()->Now()));
   275 					if(err!=KErrNone)
   276 						{
   277 						AnnotateSpriteRedrawEnd(*iWsWin, *sprite);
   278 						anim->Panic(EWservPanicAnimLeave);
   279 						return;
   280 						}
   281 					}
   282 				
   283 				//...call Redraw on the sprite
   284 				if (hasRedrawBegun)
   285 				    {
   286                     aGc->Reset();
   287 				    }
   288 				sprite->Redraw(aGc, redrawRegion);
   289 				
   290 				if (hasRedrawBegun)
   291 				    {
   292                     AnnotateSpriteRedrawEnd(*iWsWin, *sprite);
   293 				    }
   294 				}
   295 			}
   296 		redrawRegion.Close();
   297 		}
   298 	}
   299 
   300 TBool CWsWindowRedraw::Contains(const TArray<TGraphicDrawerId>& /*aDrawers*/,const TRegion& aRegion) const
   301 	{
   302 	// if in doubt, assume we do
   303 	return !aRegion.IsEmpty();
   304 	}
   305 
   306 TInt CWsWindowRedraw::DrawBackgroundColor(const TRegion& aRegion, TBool	aDoFillColor)
   307 	{
   308 	if (BackColor().Alpha() == 0 && !HasElement())
   309 		return KErrNone;
   310 	
   311 	if(aRegion.IsEmpty())
   312 		return KErrNone;
   313 	
   314 	TRect winAbs(CliWin()->AbsRect());	//fill size for background color fill
   315 	TRect surfaceAbs(0,0,0,0);			//fill size for background surface fill - initially disabled
   316 	
   317 	if (HasElement())
   318 		{
   319 		TBackgroundAttributes* backgroundAttributes = CliWin()->Screen()->WindowElements().FindBackgroundElement(*CliWin());
   320 		WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
   321 		
   322 		if (backgroundAttributes->iElement)
   323 			{
   324 			if (backgroundAttributes->ExplicitExtent())
   325 				{
   326 				backgroundAttributes->iElement->GetDestinationRectangle(surfaceAbs);
   327 				surfaceAbs.Intersection(winAbs);
   328 				if (surfaceAbs==winAbs)
   329 					{
   330 					winAbs.iBr.iX=winAbs.iTl.iX;	//disable background color fill
   331 					}
   332 				}
   333 			else
   334 				{
   335 				surfaceAbs=winAbs;
   336 				winAbs.iBr.iX=winAbs.iTl.iX;	//disable background color fill
   337 				}
   338 			}
   339 		if (!aDoFillColor)
   340 			{
   341 			winAbs.iBr.iX=winAbs.iTl.iX;	//disable background color fill
   342 			}
   343 		}
   344 	
   345 	CPlaybackGc* playback = CPlaybackGc::Instance();
   346 	MWsGraphicsContext* gc = static_cast<MWsGraphicsContext*>(playback->ResolveObjectInterface(KMWsGraphicsContext));
   347 	gc->SetClippingRegion(aRegion);
   348 	gc->SetBrushStyle(MWsGraphicsContext::ESolidBrush);
   349 	gc->SetPenStyle(MWsGraphicsContext::ENullPen);
   350 	TInt err = KErrNone;
   351 	if (!winAbs.IsEmpty())
   352 		{
   353 		gc->SetBrushColor(BackColor());
   354 		gc->DrawRect(winAbs);
   355 		}
   356 	if (!surfaceAbs.IsEmpty())
   357 		{
   358 		gc->SetDrawMode(MWsGraphicsContext::EDrawModeWriteAlpha);
   359 		gc->SetBrushColor(TRgb(0,0,0,0));
   360 		gc->DrawRect(surfaceAbs);
   361 		gc->SetBrushColor(BackColor());		//leave in a sensible state
   362 		gc->SetDrawMode(MWsGraphicsContext::EDrawModePEN);
   363 		}
   364 	gc->ResetClippingRegion();
   365 	return err;
   366 	}
   367 
   368 TBool CWsWindowRedraw::ReleaseMemory(MWsMemoryRelease::TMemoryReleaseLevel)
   369 	{
   370 	return EFalse;
   371 	}
   372 
   373 void CWsWindowRedraw::VisibleRegionChange()
   374 	{
   375 	}
   376 
   377 TBool CWsWindowRedraw::ReadyToDraw() const
   378 	{
   379 	return ETrue;
   380 	}
   381 
   382 TBool CWsWindowRedraw::RedrawingInProgress() const
   383 	{
   384 	return EFalse;	
   385 	}
   386 
   387 void CWsWindowRedraw::WindowClosing()
   388 	{
   389 	ReleaseBackgroundElement();
   390 	}
   391 
   392 TBool CWsWindowRedraw::HasDsaElement() const
   393 	{
   394 	TBool hasDsaElement = EFalse;
   395 
   396 	if (HasElement())
   397 		{
   398 		CWsClientWindow* cliWin = CliWin();
   399 		CWindowElementSet& set = cliWin->Screen()->WindowElements();
   400 		TBackgroundAttributes* backgroundAttributes = set.FindBackgroundElement(*cliWin);
   401 		WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
   402 
   403 		if (backgroundAttributes->iElement)
   404 			{
   405 			MWsElement& element = *(backgroundAttributes->iElement);
   406 			hasDsaElement = (element.ConnectedSurface() == cliWin->Screen()->DsaSurface());
   407 			}
   408 		}
   409 
   410 	return hasDsaElement;
   411 	}
   412 
   413 void CWsWindowRedraw::SetDsaElementL()
   414 	{
   415 	TRect extent(TPoint(0,0), WsWin()->Screen()->DSASizeInPixels());
   416 	MWsDisplayMapping *dispMap = WsWin()->Screen()->DisplayMapping();
   417 	TRect extentOut;
   418 	TRect extentInDSA;
   419 	if(dispMap)
   420 		{
   421 		dispMap->MapCoordinates(EDirectScreenAccessSpace,extent,EApplicationSpace,extentOut);
   422 		//DSA extent in application space intersects window extent in application space
   423 		extentOut.Intersection(WsWin()->FullRect());		
   424 		if(extentOut.IsEmpty())
   425 			{
   426 			extentOut.SetRect(0,0,0,0);
   427 			}
   428 		//use DSA coordinates to determine the viewport
   429 		dispMap->MapCoordinates(EApplicationSpace, extentOut, EDirectScreenAccessSpace, extentInDSA);
   430 		}
   431 	else
   432 		{
   433 		extentOut = extent;
   434 		extentInDSA = extent;
   435 		extentOut.Intersection(WsWin()->FullRect());
   436 		}
   437 	if (!HasDsaElement())
   438 		{
   439 		WsWin()->Screen()->ClearDsaSurface(extent, BackColor());
   440 		}
   441 
   442 	TSurfaceConfiguration sc;
   443 	sc.SetSurfaceId(WsWin()->Screen()->DsaSurface());
   444 	sc.SetExtent(extentOut.Size());
   445 	sc.SetViewport(extentInDSA);
   446 
   447 	SetBackgroundSurfaceL(sc, ETrue, ETrue);
   448 	}
   449 
   450 TBackgroundAttributes& CWsWindowRedraw::AcquireBackgroundElementL()
   451 	{
   452 	// Only client windows can have elements set
   453 	WS_ASSERT_DEBUG(iWsWin->WinType() == EWinTypeClient,EWsPanicWindowType);
   454 	CWsClientWindow* cliWin = static_cast<CWsClientWindow*>(iWsWin);
   455 	CScreen* screen = cliWin->Screen();
   456 	WS_ASSERT_DEBUG(screen,EWsPanicNoScreen);
   457 
   458 	CWindowElementSet& set = screen->WindowElements();
   459 	SetHasElement(EFalse);
   460 	TBackgroundAttributes& backgroundAttributes = set.AcquireBackgroundElementL(*cliWin);
   461 	MWsElement& element = *(backgroundAttributes.iElement);
   462 	element.SetGlobalAlpha(cliWin->IsVisible() ? 0xFF : 0);
   463 	SetHasElement(ETrue);
   464 	screen->ElementAdded();
   465 	
   466 	return backgroundAttributes;
   467 	}
   468 
   469 void CWsWindowRedraw::SetBackgroundSurfaceL(const TSurfaceId& aSurface)
   470 	{
   471 	if (aSurface.Type() == TSurfaceId::EScreenSurface || aSurface.IsNull())
   472 		{
   473 		OwnerPanic(EWservPanicInvalidSurface);
   474 		}
   475 
   476 	CWsClientWindow* cliWin = CliWin();
   477 	CScreen* screen = cliWin->Screen();
   478 	CWindowElementSet& set = screen->WindowElements();
   479 	TBackgroundAttributes& backgroundAttributes = AcquireBackgroundElementL();
   480 	MWsElement& element = *(backgroundAttributes.iElement);
   481     TInt err = set.RegisterSurface(aSurface);
   482     if (err != KErrNone)
   483         {
   484         ReleaseBackgroundElement();
   485         User::Leave(err);
   486         }
   487 	err = element.ConnectSurface(aSurface); 
   488 	if (err != KErrNone)
   489 		{
   490 		set.UnregisterSurface(aSurface);
   491 		ReleaseBackgroundElement();
   492 		User::Leave(err);
   493 		}
   494 
   495 	TRect winExtent = cliWin->FullRect();
   496 	element.SetDestinationRectangle(winExtent);
   497 	
   498 	// By default Element's source rectangle is set to its surface rectangle
   499 	TRect srcRect;
   500 	element.GetSourceRectangle(srcRect);
   501 	cliWin->SetOriginalSrcElementRect(srcRect);
   502 	cliWin->SetOriginalDestElementRect(winExtent);
   503 
   504 	SetMayContainElementFlags();
   505 	
   506 	MWsWindowTreeObserver* windowTreeObserver = Screen()->WindowTreeObserver();
   507 	if (windowTreeObserver)
   508 		{
   509 		windowTreeObserver->ElementAdded(*iWsWin, element);
   510 		}
   511 	}
   512 
   513 void CWsWindowRedraw::SetBackgroundSurfaceL(const TSurfaceConfiguration& aConfiguration, TBool aTriggerRedraw, TBool aAllowScreenSurface)
   514 	{
   515 	if (aConfiguration.Size() < sizeof(TSurfaceConfiguration))
   516 		{
   517 		__ASSERT_COMPILE(sizeof(TSurfaceConfiguration2)==sizeof(TSurfaceConfiguration));
   518 		if (aConfiguration.Size() != sizeof(TSurfaceConfiguration1))
   519 			{
   520 			OwnerPanic(EWservPanicInvalidSurfaceConfiguration);
   521 			}
   522 		}
   523 
   524 	TSurfaceId surfaceId;
   525 	aConfiguration.GetSurfaceId(surfaceId);
   526 	if ((surfaceId.Type() == TSurfaceId::EScreenSurface && !aAllowScreenSurface) || surfaceId.IsNull())
   527 		{
   528 		OwnerPanic(EWservPanicInvalidSurface);
   529 		}
   530 
   531 	CFbsBitGc::TGraphicsOrientation tempOrientation = aConfiguration.Orientation();
   532 	__ASSERT_COMPILE(CFbsBitGc::EGraphicsOrientationNormal==0 && 
   533 			CFbsBitGc::EGraphicsOrientationRotated270 == 3);
   534 	if(tempOrientation < CFbsBitGc::EGraphicsOrientationNormal || 
   535 			tempOrientation > CFbsBitGc::EGraphicsOrientationRotated270)
   536 		{
   537 		OwnerPanic(EWservPanicInvalidSurfaceConfiguration);
   538 		}
   539 
   540 	CWsClientWindow* cliWin = CliWin();
   541 	CScreen* screen = cliWin->Screen();
   542 	__ASSERT_DEBUG(screen, Panic(EWsPanicNoScreen));
   543 	
   544 	CWindowElementSet& set = screen->WindowElements();
   545 	TBool mustRegister = ETrue;
   546 	TRect oldExtent(0,0,0,0);
   547 	MWsElement::TElementRotation oldRotation = MWsElement::EElementAntiClockwise0;
   548 	TBool oldFlip = EFalse;
   549 	TRect oldViewport(0,0,0,0);
   550 	TSurfaceId oldSurfaceId = TSurfaceId::CreateNullId();
   551 
   552 	// If a element has already been set
   553 	if (HasElement())
   554 		{
   555 		TBackgroundAttributes* backgroundAttributes = set.FindBackgroundElement(*cliWin);
   556 		WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
   557 		
   558 		if (backgroundAttributes->iElement)
   559 			{
   560 			MWsElement& element = *(backgroundAttributes->iElement);
   561 			element.GetDestinationRectangle(oldExtent);
   562 			element.GetSourceRectangle(oldViewport);
   563 			oldRotation = element.SourceRotation();
   564 			oldFlip = element.SourceFlipping();
   565 			oldSurfaceId = element.ConnectedSurface();
   566 			mustRegister = EFalse;
   567 			// If it is a different surface, flag to register the new surface
   568 			if (element.ConnectedSurface() != surfaceId)
   569 				{
   570 				mustRegister = ETrue;
   571 				}
   572 			}
   573 		}
   574 	
   575 	//the call to AcquireBackgroundElementL() will remove any existing background element
   576 	TBackgroundAttributes& backgroundAttributes = mustRegister ? 
   577 			AcquireBackgroundElementL() : *(set.FindBackgroundElement(*cliWin));
   578 	MWsElement& element = *(backgroundAttributes.iElement);
   579 	TInt err = KErrNone;
   580 	if (mustRegister)
   581 		{
   582         err = set.RegisterSurface(surfaceId);
   583         switch(err)
   584             {
   585         case KErrBadHandle:
   586             // Invalid surface IDs have to return KErrArgument
   587             err = KErrArgument;
   588             // drop through
   589         case KErrNoMemory:
   590         case KErrArgument:
   591             ReleaseBackgroundElement();
   592             User::Leave(err);
   593         case KErrNone:
   594             break;
   595         default:
   596             // No need to release layer here since session closure will do it
   597             // automatically when the client thread is panicked.
   598             OwnerPanic(EWservPanicInvalidSurface);
   599             }
   600 
   601 		err = element.ConnectSurface(surfaceId);
   602 		if (err != KErrNone)
   603 			{
   604 			set.UnregisterSurface(surfaceId);
   605 			ReleaseBackgroundElement();	//Releasing new empty element
   606 			User::Leave(err);
   607 			}
   608 
   609 		if (screen->DsaSurface() == surfaceId)
   610 			{
   611 	         TUint32 flags; 
   612 	         element.GetRenderStageFlags(flags);
   613 	         flags |= MWsElement::EElementIsDirectlyRenderedUserInterface;
   614 	         element.SetRenderStageFlags(flags);
   615 			}
   616 		}
   617 
   618 	SetHasElement(ETrue);		//set element flag
   619 	SetMayContainElementFlags();
   620 	
   621 	err = CWindowElement::SetElement(element,aConfiguration,ETrue);	//set viewport and orientation
   622 	if (err == KErrArgument)
   623 		{
   624 		OwnerPanic(EWservPanicInvalidSurfaceConfiguration);
   625 		}
   626     TRect srcRect;
   627     aConfiguration.GetViewport(srcRect);
   628     if (!srcRect.IsEmpty())
   629         backgroundAttributes.SetExplicitViewPort();
   630     element.GetSourceRectangle(srcRect);
   631     cliWin->SetOriginalSrcElementRect(srcRect);
   632 
   633 	//Set Extent
   634 	TRect newExtent;
   635 	aConfiguration.GetExtent(newExtent);
   636 	SetElementExtentL(newExtent, backgroundAttributes);
   637 	cliWin->SetOriginalDestElementRect(newExtent);
   638 	
   639 	MWsWindowTreeObserver* windowTreeObserver = Screen()->WindowTreeObserver();
   640 	if (windowTreeObserver && mustRegister)
   641 		{
   642 		windowTreeObserver->ElementAdded(*iWsWin, element);
   643 		}
   644 	
   645 	//If set, redraw
   646 	if (aTriggerRedraw)
   647 		{
   648 		TRect newViewport;
   649 		aConfiguration.GetViewport(newViewport);
   650 		CFbsBitGc::TGraphicsOrientation orientation = aConfiguration.Orientation();
   651 		MWsElement::TElementRotation newRotation = GcToElementRotation(orientation);
   652 		TBool newFlip = aConfiguration.Flip();
   653 		
   654 		//The following parameter guarantees that update will be scheduled. 
   655 		//This will trigger the composition.
   656 		TBool alwaysScheduleUpdate = (oldSurfaceId != surfaceId) || 
   657 									(oldExtent != newExtent) ||
   658 									(oldViewport != newViewport) ||
   659 									(oldRotation != newRotation)||
   660 									(oldFlip != newFlip);
   661 									
   662 		ElementRedraw(oldExtent,newExtent,alwaysScheduleUpdate);
   663 		}
   664 	}
   665 
   666 /**
   667 Sets the EMayContainElement flag for parent window.
   668 Sets the flag for all ancestor windows.
   669 **/
   670 void CWsWindowRedraw::SetMayContainElementFlags()
   671 	{
   672 	CWsWindowBase* parent = CliWin()->BaseParent();
   673 	TInt type = parent->WinType();
   674 	while(type ==EWinTypeClient)
   675 			{
   676 			CWsClientWindow* win = static_cast<CWsClientWindow*>(parent);
   677 			win->Redraw()->iStateFlags |= EMayContainElement;
   678 			parent=parent->BaseParent();
   679 			type = parent->WinType();
   680 			}
   681 	}
   682 
   683 void CWsWindowRedraw::SetElementExtentL(TRect& aNewExtent, TBackgroundAttributes& aAttributes)
   684 	{
   685 	CWsClientWindow* cliWin = CliWin();
   686 	MWsElement& element = *(aAttributes.iElement);
   687 	if (aNewExtent.IsEmpty())
   688 		{
   689 		aNewExtent = cliWin->FullRect();
   690 		aAttributes.SetExplicitExtent(EFalse);
   691 		}
   692 	else
   693 		{
   694 		TRect tempWindowPosition = cliWin->FullRect();		//get window absolute coordinates
   695 		aNewExtent.Move(tempWindowPosition.iTl);				//shift user defined extent to absolute coordinates
   696 		aAttributes.SetExplicitExtent(ETrue);
   697 		}
   698 	element.SetDestinationRectangle(aNewExtent);
   699 	}
   700 
   701 void CWsWindowRedraw::ElementRedraw(const TRect& aOldExtent, const TRect& aNewExtent, TBool aAlwaysScheduleUpdate)
   702 	{
   703 	if (!aOldExtent.IsEmpty())
   704 		{
   705 		//If the previous extent was different
   706 		if (aOldExtent != aNewExtent)
   707 			{
   708 			STACK_REGION tempRegion;
   709 			tempRegion.AddRect(aOldExtent);
   710 			tempRegion.AddRect(aNewExtent);
   711 
   712 			//Calculate the difference between
   713 			TRect tempRect = aOldExtent;
   714 			tempRect.Intersection(aNewExtent);			//intersect both regions
   715 			tempRegion.SubRect(tempRect);		//cut unaltered region
   716 			Screen()->ScheduleRegionUpdate(&tempRegion);
   717 
   718 			tempRegion.Close();
   719 			}
   720 		else
   721 			{
   722 			if(aAlwaysScheduleUpdate)
   723 				{
   724 				TTimeIntervalMicroSeconds interval(0);
   725 				Screen()->ScheduleRender(interval);
   726 				}
   727 			}
   728 		}
   729 	else
   730 		{
   731 		TRegionFix<1> region(aNewExtent);
   732 		Screen()->ScheduleRegionUpdate(&region);
   733 		}
   734 	}
   735 
   736 void CWsWindowRedraw::RemoveBackgroundSurface(TBool aTriggerRedraw)
   737 	{
   738 	if (HasElement())
   739 		{
   740 		TBackgroundAttributes* backgroundAttributes = CliWin()->Screen()->
   741 				WindowElements().FindBackgroundElement(*CliWin());
   742 		WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
   743 		if (backgroundAttributes->iElement)
   744 			{
   745 			RemoveBackgroundElement(aTriggerRedraw);
   746 			}
   747 		}
   748 	}
   749 
   750 void CWsWindowRedraw::RemoveBackgroundElement(TBool aTriggerRedraw)
   751 	{
   752 	CWsClientWindow* cliWin = CliWin();
   753 	CScreen* screen = cliWin->Screen();
   754 	TRect tempRect;
   755 	if (aTriggerRedraw)
   756 		{
   757 		TBackgroundAttributes* backgroundAttributes = screen->WindowElements().FindBackgroundElement(*CliWin());
   758 		WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
   759 		if (backgroundAttributes->ExplicitExtent())
   760 			{
   761 			backgroundAttributes->iElement->GetDestinationRectangle(tempRect);
   762 			backgroundAttributes->SetExplicitExtent(EFalse);
   763 			}
   764 		else
   765 			{
   766 			tempRect = cliWin->FullRect();
   767 			}
   768 		}
   769 	ReleaseBackgroundElement();
   770     if (aTriggerRedraw)
   771         {
   772         if (screen->ChangeTracking())
   773             {
   774             TTimeIntervalMicroSeconds interval(0);
   775             screen->ScheduleRender(interval);
   776             }
   777         else
   778             {
   779             TRegionFix<1> region(tempRect);
   780             screen->ScheduleRegionUpdate(&region);
   781             }
   782         }
   783 	}
   784 
   785 void CWsWindowRedraw::GetBackgroundSurfaceL(TSurfaceConfiguration& aConfiguration)
   786 	{
   787 	if (aConfiguration.Size() < sizeof(TSurfaceConfiguration))
   788 		{
   789 		__ASSERT_COMPILE(sizeof(TSurfaceConfiguration2)==sizeof(TSurfaceConfiguration));
   790 		if (aConfiguration.Size() != sizeof(TSurfaceConfiguration1))
   791 			{
   792 			OwnerPanic(EWservPanicInvalidSurfaceConfiguration);
   793 			}
   794 		}
   795 	
   796 	CWsClientWindow* cliWin = CliWin();
   797 	TBackgroundAttributes* backgroundAttributes = NULL;
   798 
   799 	if (HasElement())
   800 		{
   801 		backgroundAttributes = cliWin->Screen()->WindowElements().FindBackgroundElement(*cliWin);
   802 		WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
   803 		if (!backgroundAttributes->iElement)
   804 			{
   805 			User::Leave(KErrNotFound);
   806 			}
   807 		}
   808 	else
   809 		{
   810 		User::Leave(KErrNotFound);
   811 		}
   812 
   813 	MWsElement& element = *(backgroundAttributes->iElement);
   814 
   815 	TInt errCode=CWindowElementSet::GetConfiguration(aConfiguration,element);
   816 
   817 	// Get source rect
   818 	if (errCode>=KErrNone)
   819 	    {
   820 	    if (!backgroundAttributes->ExplicitViewPort())
   821 	        {
   822 	        aConfiguration.SetViewport(TRect());
   823 	        }
   824 	    else
   825 	        {
   826 	        TRect tempExtent = cliWin->GetOriginalSrcElementRect();
   827 	        aConfiguration.SetViewport(tempExtent);
   828 	        }
   829 	    }
   830 
   831 	//Convert and copy extent
   832 	if (errCode>=KErrNone)
   833 		{
   834 		if (!backgroundAttributes->ExplicitExtent())
   835 			{
   836 			aConfiguration.SetExtent(TRect());
   837 			}
   838 		else	//translate to window coordinates
   839 			{
   840 			TRect tempExtent = cliWin->GetOriginalDestElementRect();
   841 			tempExtent.Move(-cliWin->Origin());
   842 			aConfiguration.SetExtent(tempExtent);
   843 			}
   844 		}
   845 	}
   846 
   847 void CWsWindowRedraw::ReleaseBackgroundElement()
   848 	{
   849 	if (HasElement())
   850 		{
   851 		CWsClientWindow* cliWin = CliWin();
   852 		CScreen* screen = cliWin->Screen();
   853 		screen->WindowElements().ReleaseBackgroundElement(*cliWin, ETrue);
   854 		screen->ElementRemoved();
   855 		}
   856 	}
   857 
   858 //
   859 // Blank window //
   860 //
   861 
   862 CWsBlankWindow::CWsBlankWindow(CWsWindow *aWin) : CWsWindowRedraw(aWin), iColor(iWsWin->RootWindow()->DefaultBackgroundColor()), iNoColor(EFalse)
   863 	{
   864 	}
   865 
   866 CWsBlankWindow::~CWsBlankWindow()
   867 	{
   868 	}
   869 
   870 void CWsBlankWindow::ConstructL()
   871 	{
   872 	CWsWindowRedraw::ConstructL();
   873 	if (Screen()->ChangeTracking())
   874 		{
   875 		STACK_REGION dirtyRegion;
   876 		dirtyRegion.Copy(iWsWin->WindowArea());
   877 		dirtyRegion.Offset(-iWsWin->Origin());
   878 		iWsWin->AddDirtyWindowRegion(dirtyRegion);
   879 		dirtyRegion.Close();
   880 		}
   881 	}
   882 
   883 void CWsBlankWindow::SetColor(TRgb aColor)
   884 	{
   885 	iColor=aColor;
   886 	iNoColor=EFalse;
   887 	if (Screen()->ChangeTracking())
   888 		{
   889 		STACK_REGION dirtyRegion;
   890 		dirtyRegion.Copy(iWsWin->WindowArea());
   891 		dirtyRegion.Offset(-iWsWin->Origin());
   892 		iWsWin->AddDirtyWindowRegion(dirtyRegion);
   893 		dirtyRegion.Close();
   894 
   895 		if (iWsWin->IsActive() && iWsWin->IsVisible())
   896 			{
   897 			Screen()->ScheduleWindow(iWsWin);
   898 			}
   899 		}
   900 	else
   901 		{
   902 		Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
   903 		}
   904 	}
   905 
   906 TBool CWsBlankWindow::CommandL(TInt aOpcode, TWsWinCmdUnion &aCmd)
   907 	{
   908 	switch(aOpcode)
   909 		{
   910 		case EWsWinOpSetBackgroundSurface:
   911 			SetBackgroundSurfaceL(*aCmd.Surface);
   912 			break;
   913 		case EWsWinOpSetBackgroundSurfaceConfig:
   914 			SetBackgroundSurfaceL(aCmd.SurfaceConfigurationAndTrigger->surfaceConfig, aCmd.SurfaceConfigurationAndTrigger->triggerRedraw, EFalse);
   915 			break;
   916 		case EWsWinOpRemoveBackgroundSurface:
   917 			RemoveBackgroundSurface(*aCmd.Bool);
   918 			break;
   919 		case EWsWinOpGetBackgroundSurfaceConfig:
   920 			{
   921 			TSurfaceConfiguration tempConfiguration = *aCmd.SurfaceConfiguration;
   922 			GetBackgroundSurfaceL(tempConfiguration);
   923 			TInt tempSize = aCmd.SurfaceConfiguration->Size();
   924 			if (sizeof(TSurfaceConfiguration)<tempSize)
   925 				tempSize = sizeof(TSurfaceConfiguration);
   926 			CWsClient::ReplyBuf(&tempConfiguration,tempSize);
   927 			}
   928 			break;
   929 		case EWsWinOpSetColor:
   930 			SetColor(*aCmd.rgb);
   931 			break;
   932 		case EWsWinOpSetNoBackgroundColor:
   933 			SetBackgroundClear();
   934 			break;
   935 		default:
   936 			return(EFalse);
   937 		}
   938 	return(ETrue);
   939 	}
   940 
   941 TRgb CWsBlankWindow::BackColor() const
   942 	{
   943 	return(iColor);
   944 	}
   945 
   946 TBool CWsBlankWindow::GetRedrawRect(TRect &) const
   947 	{
   948 	if (!iNoColor || iWsWin->iAnimList)
   949 		iWsWin->Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
   950 	return(EFalse);
   951 	}
   952 
   953 TBool CWsBlankWindow::NeedsRedraw() const
   954 	{
   955 	return(EFalse);
   956 	}
   957 
   958 void CWsBlankWindow::DrawWindow()
   959 	{
   960 	if ((!iNoColor)||HasElement())
   961 		{
   962 		DrawBackgroundColor(*iRedrawRegion,!iNoColor);
   963 		}
   964 	}