1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nga/SERVER/openwfc/wnredraw.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,964 @@
1.4 +// Copyright (c) 1995-2010 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 +// Window redraw code, three sorts of redrawing are supported
1.18 +// Sending a redraw message to the client (see redrawmsgwindow.cpp)
1.19 +// Drawing from backup bitmap
1.20 +// Simply clearing the window
1.21 +//
1.22 +//
1.23 +
1.24 +#include "wnredraw.h"
1.25 +#include "server.h"
1.26 +#include "playbackgc.h"
1.27 +#include "wstop.h"
1.28 +#include "ANIM.H"
1.29 +#include "EVQUEUE.H"
1.30 +#include <s32mem.h>
1.31 +#include <gdi.h>
1.32 +#include "panics.h"
1.33 +#include "inifile.h"
1.34 +#include "rootwin.h"
1.35 +#include "EVENT.H"
1.36 +#include "wstypes.h"
1.37 +#include <graphics/surface.h>
1.38 +#include <graphics/wselement.h>
1.39 +#include <graphics/wsscreendevice.h>
1.40 +#include "windowelementset.h"
1.41 +
1.42 +struct TFadingParams
1.43 + {
1.44 + TUint8 blackMap;
1.45 + TUint8 whiteMap;
1.46 + };
1.47 +
1.48 +CWsWindowRedraw::CWsWindowRedraw(CWsWindow *aWin) : iWsWin(aWin)
1.49 + {
1.50 + }
1.51 +
1.52 +CWsWindowRedraw::~CWsWindowRedraw()
1.53 + {
1.54 + if (iWsWin->WsOwner())
1.55 + {
1.56 + iWsWin->WsOwner()->RedrawQueue()->RemoveInvalid(this);
1.57 + }
1.58 + if (HasElement())
1.59 + {
1.60 + iWsWin->Screen()->WindowElements().ReleaseAllElements(*CliWin());
1.61 + }
1.62 + }
1.63 +
1.64 +void CWsWindowRedraw::ConstructL()
1.65 + {
1.66 + }
1.67 +
1.68 +const TRegion& CWsWindowRedraw::InvalidArea() const
1.69 + {
1.70 + return(nullRegion);
1.71 + }
1.72 +
1.73 +const TRegion &CWsWindowRedraw::BaseDrawRegion() const
1.74 + {
1.75 + return(iWsWin->VisibleRegion());
1.76 + }
1.77 +
1.78 +void CWsWindowRedraw::ClipInvalidRegion(const TRect &)
1.79 + {
1.80 + }
1.81 +
1.82 +void CWsWindowRedraw::Resize(const TSize &, const TSize &)
1.83 + {
1.84 + }
1.85 +
1.86 +void CWsWindowRedraw::SetReply(TInt aReply)
1.87 + {
1.88 + iWsWin->WsOwner()->SetReply(aReply);
1.89 + }
1.90 +
1.91 +void CWsWindowRedraw::OwnerPanic(TClientPanic aPanic)
1.92 + {
1.93 + iWsWin->OwnerPanic(aPanic);
1.94 + }
1.95 +
1.96 +CWsBackedUpWindow *CWsWindowRedraw::Backup() const
1.97 + {
1.98 + return(NULL);
1.99 + }
1.100 +
1.101 +void CWsWindowRedraw::Scroll(const TRect &, const TPoint &,const TRect &)
1.102 + {
1.103 + }
1.104 +
1.105 +void CWsWindowRedraw::UpdateAnimArea()
1.106 + {
1.107 + }
1.108 +
1.109 +void CWsWindowRedraw::PrepareForResizeL(const TSize& /* aNewSize */, TSize& /* aOldSize */)
1.110 + {
1.111 + }
1.112 +
1.113 +TBool CWsWindowRedraw::DrawCommand(CWsGc*,const TAny*)
1.114 + {
1.115 + return ETrue;
1.116 + }
1.117 +
1.118 +void CWsWindowRedraw::GcAttributeChange(CWsGc*,const TAny*)
1.119 + {
1.120 + }
1.121 +
1.122 +void CWsWindowRedraw::GcDeactivate(CWsGc*)
1.123 + {
1.124 + }
1.125 +
1.126 +CFbsDevice* CWsWindowRedraw::OutputDevice() const
1.127 + {
1.128 + return NULL;
1.129 + }
1.130 +
1.131 +void CWsWindowRedraw::ClientExposing()
1.132 + {
1.133 + }
1.134 +
1.135 +void CWsWindowRedraw::ClearRedrawStore(TBool)
1.136 + {}
1.137 +
1.138 +void CWsWindowRedraw::PreDrawWindow(MWsGraphicsContext* aGc, const TRegion& aWindowRegion)
1.139 + {
1.140 + WS_ASSERT_DEBUG(iRedrawRegion == NULL, EWsPanicScheduledRedraw);
1.141 + iRedrawRegion = &aWindowRegion;
1.142 + CPlaybackGc::Instance()->SetTargetRegion(iRedrawRegion);
1.143 + CWsClient::iCurrentCommand.iOpcode=0;
1.144 + CPlaybackGc::Instance()->Activate(CliWin(), aGc, iRedrawRegion);
1.145 + }
1.146 +
1.147 +void CWsWindowRedraw::PostDrawWindow(MWsGraphicsContext* aGc, const TRegion& aWindowChildNodeRegion)
1.148 + {
1.149 + WS_ASSERT_DEBUG(iRedrawRegion, EWsPanicScheduledRedraw);
1.150 + CPlaybackGc::Instance()->Deactivate();
1.151 + CPlaybackGc::Instance()->SetTargetRegion(NULL);
1.152 +
1.153 + if(!Screen()->ChangeTracking())
1.154 + {
1.155 + DoFade(*iRedrawRegion);
1.156 + }
1.157 +
1.158 + AnnotateWindowRedrawEnd(*iWsWin);
1.159 +
1.160 + DrawWindowAnims(aGc, aWindowChildNodeRegion);
1.161 + DrawCursorAndSprites(aGc, aWindowChildNodeRegion);
1.162 + iRedrawRegion = 0;
1.163 + }
1.164 +
1.165 +void CWsWindowRedraw::Fade(MWsGraphicsContext * aGc, const TRegion& aRegion)
1.166 + {
1.167 + LOG_WINDOW_FADE_START(WsWin());
1.168 + AnnotateWindowRedrawStart(*iWsWin, aRegion);
1.169 +
1.170 + aGc->Reset();
1.171 + DoFade(aRegion);
1.172 +
1.173 + AnnotateWindowRedrawEnd(*iWsWin);
1.174 + LOG_WINDOW_FADE_END(WsWin());
1.175 + }
1.176 +
1.177 +void CWsWindowRedraw::DoFade(const TRegion& aRegion)
1.178 + {
1.179 + if( CWsTop::IsFadeEnabled() && iWsWin && iWsWin->FadeCount()>0 && !(iWsWin->IsNonFading()) && !(iWsWin->FadableRegion().IsEmpty()) && !(iWsWin->IsDSAHost()) )
1.180 + {
1.181 + MWsFader* fader = static_cast<MWsFader*>(iWsWin->Screen()->ResolveObjectInterface(KMWsFader));
1.182 + if(fader)
1.183 + {
1.184 + TFadingParams parameters;
1.185 + iWsWin->GetFadingParams(parameters.blackMap,parameters.whiteMap);
1.186 + TPckgBuf<TFadingParams> buf(parameters);
1.187 + fader->SetFadingParameters(buf);
1.188 + // Only fade the region that hasn't been faded before
1.189 + STACK_REGION fdRgn;
1.190 + fdRgn.Copy( aRegion );
1.191 + fdRgn.Intersect( iWsWin->FadableRegion() );
1.192 + if(!fdRgn.CheckError())
1.193 + {
1.194 + fader->FadeArea( fdRgn );
1.195 + LOG_WINDOW_FADE_REGION(&fdRgn);
1.196 + }
1.197 + fdRgn.Close();
1.198 + }
1.199 + }
1.200 + }
1.201 +
1.202 +void CWsWindowRedraw::DrawWindowAnims(MWsGraphicsContext * aGc, const TRegion& aRegion)
1.203 + {
1.204 + if (iWsWin->iAnimList)
1.205 + {
1.206 + // If an anim panics, it will leave and set the panic flag on the client
1.207 + // The client itself won't actually panic yet, and we don't want to leave from here.
1.208 + TRAP_IGNORE(DrawWindowAnimsL(aGc, aRegion));
1.209 + }
1.210 + }
1.211 +
1.212 +void CWsWindowRedraw::DrawWindowAnimsL(MWsGraphicsContext * aGc, const TRegion& aRegion)
1.213 + {
1.214 + for (CWsAnim * anim = iWsWin->iAnimList; anim; anim = anim->Next())
1.215 + {
1.216 + AnnotateWindowAnimRedrawStart(*iWsWin, *anim, aRegion);
1.217 +
1.218 + //Animate and redraw
1.219 + TRAPD(err,anim->RedrawWindowAnimL(Screen()->Now(), aGc, &aRegion));
1.220 + if(err!=KErrNone)
1.221 + {
1.222 + AnnotateWindowAnimRedrawEnd(*iWsWin, *anim);
1.223 + anim->Panic(EWservPanicAnimLeave);
1.224 + return;
1.225 + }
1.226 +
1.227 + AnnotateWindowAnimRedrawEnd(*iWsWin, *anim);
1.228 + }
1.229 + }
1.230 +
1.231 +void CWsWindowRedraw::DrawCursorAndSprites(MWsGraphicsContext * aGc, const TRegion& aRegion)
1.232 + {
1.233 + // Draw standard text cursor if required
1.234 + RWsTextCursor* const cursor = CWsTop::CurrentTextCursor();
1.235 + if (!iWsWin->Screen()->ChangeTracking() && cursor && cursor->Win() == iWsWin && cursor->IsStandardCursorActive())
1.236 + {
1.237 + // Standard text cursor is active on this window
1.238 + const TBool flashing = cursor->IsFlashing();
1.239 + TFlashState flashState = EFlashOn;
1.240 + if (flashing)
1.241 + {
1.242 + flashState = cursor->CurrentCursorFlashState();
1.243 + }
1.244 + if (flashState == EFlashOn)
1.245 + {
1.246 + // Cursor should be visible, so draw it
1.247 + cursor->Draw(aRegion);
1.248 + }
1.249 + if (flashing)
1.250 + {
1.251 + // Reschedule to flash the standard cursor on or off
1.252 + Screen()->ScheduleAnimation(ETextCursor, cursor->RectRelativeToScreen(), Screen()->SpriteManager()->NextCursorFlashStateChange(), 0, 0, iWsWin);
1.253 + }
1.254 + }
1.255 +
1.256 + for (CWsSpriteBase * sprite = iWsWin->iSpriteList; sprite; sprite = sprite->Next())
1.257 + {
1.258 + TBool hasRedrawBegun = EFalse;
1.259 + STACK_REGION redrawRegion;
1.260 + sprite->CalcRedrawRegion(aRegion, redrawRegion);
1.261 + if(redrawRegion.CheckError() || !redrawRegion.IsEmpty())
1.262 + {
1.263 + if (sprite->IsFlashingEnabled() || sprite->IsDirty() || sprite->HasAnimation())
1.264 + {
1.265 + if (sprite->IsDirty() || sprite->HasAnimation())
1.266 + {
1.267 + AnnotateSpriteRedrawStart(*iWsWin, *sprite, redrawRegion);
1.268 + hasRedrawBegun = ETrue;
1.269 + }
1.270 +
1.271 + if(sprite->HasAnimation())
1.272 + {
1.273 + CWsAnim* anim = static_cast<CWsSprite*>(sprite)->iAnim;
1.274 + WS_ASSERT_DEBUG(anim,EWsPanicAnim);
1.275 +
1.276 + //Animate and...
1.277 + TRAPD(err, anim->AnimateSpriteAnimL(Screen()->Now()));
1.278 + if(err!=KErrNone)
1.279 + {
1.280 + AnnotateSpriteRedrawEnd(*iWsWin, *sprite);
1.281 + anim->Panic(EWservPanicAnimLeave);
1.282 + return;
1.283 + }
1.284 + }
1.285 +
1.286 + //...call Redraw on the sprite
1.287 + if (hasRedrawBegun)
1.288 + {
1.289 + aGc->Reset();
1.290 + }
1.291 + sprite->Redraw(aGc, redrawRegion);
1.292 +
1.293 + if (hasRedrawBegun)
1.294 + {
1.295 + AnnotateSpriteRedrawEnd(*iWsWin, *sprite);
1.296 + }
1.297 + }
1.298 + }
1.299 + redrawRegion.Close();
1.300 + }
1.301 + }
1.302 +
1.303 +TBool CWsWindowRedraw::Contains(const TArray<TGraphicDrawerId>& /*aDrawers*/,const TRegion& aRegion) const
1.304 + {
1.305 + // if in doubt, assume we do
1.306 + return !aRegion.IsEmpty();
1.307 + }
1.308 +
1.309 +TInt CWsWindowRedraw::DrawBackgroundColor(const TRegion& aRegion, TBool aDoFillColor)
1.310 + {
1.311 + if (BackColor().Alpha() == 0 && !HasElement())
1.312 + return KErrNone;
1.313 +
1.314 + if(aRegion.IsEmpty())
1.315 + return KErrNone;
1.316 +
1.317 + TRect winAbs(CliWin()->AbsRect()); //fill size for background color fill
1.318 + TRect surfaceAbs(0,0,0,0); //fill size for background surface fill - initially disabled
1.319 +
1.320 + if (HasElement())
1.321 + {
1.322 + TBackgroundAttributes* backgroundAttributes = CliWin()->Screen()->WindowElements().FindBackgroundElement(*CliWin());
1.323 + WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
1.324 +
1.325 + if (backgroundAttributes->iElement)
1.326 + {
1.327 + if (backgroundAttributes->ExplicitExtent())
1.328 + {
1.329 + backgroundAttributes->iElement->GetDestinationRectangle(surfaceAbs);
1.330 + surfaceAbs.Intersection(winAbs);
1.331 + if (surfaceAbs==winAbs)
1.332 + {
1.333 + winAbs.iBr.iX=winAbs.iTl.iX; //disable background color fill
1.334 + }
1.335 + }
1.336 + else
1.337 + {
1.338 + surfaceAbs=winAbs;
1.339 + winAbs.iBr.iX=winAbs.iTl.iX; //disable background color fill
1.340 + }
1.341 + }
1.342 + if (!aDoFillColor)
1.343 + {
1.344 + winAbs.iBr.iX=winAbs.iTl.iX; //disable background color fill
1.345 + }
1.346 + }
1.347 +
1.348 + CPlaybackGc* playback = CPlaybackGc::Instance();
1.349 + MWsGraphicsContext* gc = static_cast<MWsGraphicsContext*>(playback->ResolveObjectInterface(KMWsGraphicsContext));
1.350 + gc->SetClippingRegion(aRegion);
1.351 + gc->SetBrushStyle(MWsGraphicsContext::ESolidBrush);
1.352 + gc->SetPenStyle(MWsGraphicsContext::ENullPen);
1.353 + TInt err = KErrNone;
1.354 + if (!winAbs.IsEmpty())
1.355 + {
1.356 + gc->SetBrushColor(BackColor());
1.357 + gc->DrawRect(winAbs);
1.358 + }
1.359 + if (!surfaceAbs.IsEmpty())
1.360 + {
1.361 + gc->SetDrawMode(MWsGraphicsContext::EDrawModeWriteAlpha);
1.362 + gc->SetBrushColor(TRgb(0,0,0,0));
1.363 + gc->DrawRect(surfaceAbs);
1.364 + gc->SetBrushColor(BackColor()); //leave in a sensible state
1.365 + gc->SetDrawMode(MWsGraphicsContext::EDrawModePEN);
1.366 + }
1.367 + gc->ResetClippingRegion();
1.368 + return err;
1.369 + }
1.370 +
1.371 +TBool CWsWindowRedraw::ReleaseMemory(MWsMemoryRelease::TMemoryReleaseLevel)
1.372 + {
1.373 + return EFalse;
1.374 + }
1.375 +
1.376 +void CWsWindowRedraw::VisibleRegionChange()
1.377 + {
1.378 + }
1.379 +
1.380 +TBool CWsWindowRedraw::ReadyToDraw() const
1.381 + {
1.382 + return ETrue;
1.383 + }
1.384 +
1.385 +TBool CWsWindowRedraw::RedrawingInProgress() const
1.386 + {
1.387 + return EFalse;
1.388 + }
1.389 +
1.390 +void CWsWindowRedraw::WindowClosing()
1.391 + {
1.392 + ReleaseBackgroundElement();
1.393 + }
1.394 +
1.395 +TBool CWsWindowRedraw::HasDsaElement() const
1.396 + {
1.397 + TBool hasDsaElement = EFalse;
1.398 +
1.399 + if (HasElement())
1.400 + {
1.401 + CWsClientWindow* cliWin = CliWin();
1.402 + CWindowElementSet& set = cliWin->Screen()->WindowElements();
1.403 + TBackgroundAttributes* backgroundAttributes = set.FindBackgroundElement(*cliWin);
1.404 + WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
1.405 +
1.406 + if (backgroundAttributes->iElement)
1.407 + {
1.408 + MWsElement& element = *(backgroundAttributes->iElement);
1.409 + hasDsaElement = (element.ConnectedSurface() == cliWin->Screen()->DsaSurface());
1.410 + }
1.411 + }
1.412 +
1.413 + return hasDsaElement;
1.414 + }
1.415 +
1.416 +void CWsWindowRedraw::SetDsaElementL()
1.417 + {
1.418 + TRect extent(TPoint(0,0), WsWin()->Screen()->DSASizeInPixels());
1.419 + MWsDisplayMapping *dispMap = WsWin()->Screen()->DisplayMapping();
1.420 + TRect extentOut;
1.421 + TRect extentInDSA;
1.422 + if(dispMap)
1.423 + {
1.424 + dispMap->MapCoordinates(EDirectScreenAccessSpace,extent,EApplicationSpace,extentOut);
1.425 + //DSA extent in application space intersects window extent in application space
1.426 + extentOut.Intersection(WsWin()->FullRect());
1.427 + if(extentOut.IsEmpty())
1.428 + {
1.429 + extentOut.SetRect(0,0,0,0);
1.430 + }
1.431 + //use DSA coordinates to determine the viewport
1.432 + dispMap->MapCoordinates(EApplicationSpace, extentOut, EDirectScreenAccessSpace, extentInDSA);
1.433 + }
1.434 + else
1.435 + {
1.436 + extentOut = extent;
1.437 + extentInDSA = extent;
1.438 + extentOut.Intersection(WsWin()->FullRect());
1.439 + }
1.440 + if (!HasDsaElement())
1.441 + {
1.442 + WsWin()->Screen()->ClearDsaSurface(extent, BackColor());
1.443 + }
1.444 +
1.445 + TSurfaceConfiguration sc;
1.446 + sc.SetSurfaceId(WsWin()->Screen()->DsaSurface());
1.447 + sc.SetExtent(extentOut.Size());
1.448 + sc.SetViewport(extentInDSA);
1.449 +
1.450 + SetBackgroundSurfaceL(sc, ETrue, ETrue);
1.451 + }
1.452 +
1.453 +TBackgroundAttributes& CWsWindowRedraw::AcquireBackgroundElementL()
1.454 + {
1.455 + // Only client windows can have elements set
1.456 + WS_ASSERT_DEBUG(iWsWin->WinType() == EWinTypeClient,EWsPanicWindowType);
1.457 + CWsClientWindow* cliWin = static_cast<CWsClientWindow*>(iWsWin);
1.458 + CScreen* screen = cliWin->Screen();
1.459 + WS_ASSERT_DEBUG(screen,EWsPanicNoScreen);
1.460 +
1.461 + CWindowElementSet& set = screen->WindowElements();
1.462 + SetHasElement(EFalse);
1.463 + TBackgroundAttributes& backgroundAttributes = set.AcquireBackgroundElementL(*cliWin);
1.464 + MWsElement& element = *(backgroundAttributes.iElement);
1.465 + element.SetGlobalAlpha(cliWin->IsVisible() ? 0xFF : 0);
1.466 + SetHasElement(ETrue);
1.467 + screen->ElementAdded();
1.468 +
1.469 + return backgroundAttributes;
1.470 + }
1.471 +
1.472 +void CWsWindowRedraw::SetBackgroundSurfaceL(const TSurfaceId& aSurface)
1.473 + {
1.474 + if (aSurface.Type() == TSurfaceId::EScreenSurface || aSurface.IsNull())
1.475 + {
1.476 + OwnerPanic(EWservPanicInvalidSurface);
1.477 + }
1.478 +
1.479 + CWsClientWindow* cliWin = CliWin();
1.480 + CScreen* screen = cliWin->Screen();
1.481 + CWindowElementSet& set = screen->WindowElements();
1.482 + TBackgroundAttributes& backgroundAttributes = AcquireBackgroundElementL();
1.483 + MWsElement& element = *(backgroundAttributes.iElement);
1.484 + TInt err = set.RegisterSurface(aSurface);
1.485 + if (err != KErrNone)
1.486 + {
1.487 + ReleaseBackgroundElement();
1.488 + User::Leave(err);
1.489 + }
1.490 + err = element.ConnectSurface(aSurface);
1.491 + if (err != KErrNone)
1.492 + {
1.493 + set.UnregisterSurface(aSurface);
1.494 + ReleaseBackgroundElement();
1.495 + User::Leave(err);
1.496 + }
1.497 +
1.498 + TRect winExtent = cliWin->FullRect();
1.499 + element.SetDestinationRectangle(winExtent);
1.500 +
1.501 + // By default Element's source rectangle is set to its surface rectangle
1.502 + TRect srcRect;
1.503 + element.GetSourceRectangle(srcRect);
1.504 + cliWin->SetOriginalSrcElementRect(srcRect);
1.505 + cliWin->SetOriginalDestElementRect(winExtent);
1.506 +
1.507 + SetMayContainElementFlags();
1.508 +
1.509 + MWsWindowTreeObserver* windowTreeObserver = Screen()->WindowTreeObserver();
1.510 + if (windowTreeObserver)
1.511 + {
1.512 + windowTreeObserver->ElementAdded(*iWsWin, element);
1.513 + }
1.514 + }
1.515 +
1.516 +void CWsWindowRedraw::SetBackgroundSurfaceL(const TSurfaceConfiguration& aConfiguration, TBool aTriggerRedraw, TBool aAllowScreenSurface)
1.517 + {
1.518 + if (aConfiguration.Size() < sizeof(TSurfaceConfiguration))
1.519 + {
1.520 + __ASSERT_COMPILE(sizeof(TSurfaceConfiguration2)==sizeof(TSurfaceConfiguration));
1.521 + if (aConfiguration.Size() != sizeof(TSurfaceConfiguration1))
1.522 + {
1.523 + OwnerPanic(EWservPanicInvalidSurfaceConfiguration);
1.524 + }
1.525 + }
1.526 +
1.527 + TSurfaceId surfaceId;
1.528 + aConfiguration.GetSurfaceId(surfaceId);
1.529 + if ((surfaceId.Type() == TSurfaceId::EScreenSurface && !aAllowScreenSurface) || surfaceId.IsNull())
1.530 + {
1.531 + OwnerPanic(EWservPanicInvalidSurface);
1.532 + }
1.533 +
1.534 + CFbsBitGc::TGraphicsOrientation tempOrientation = aConfiguration.Orientation();
1.535 + __ASSERT_COMPILE(CFbsBitGc::EGraphicsOrientationNormal==0 &&
1.536 + CFbsBitGc::EGraphicsOrientationRotated270 == 3);
1.537 + if(tempOrientation < CFbsBitGc::EGraphicsOrientationNormal ||
1.538 + tempOrientation > CFbsBitGc::EGraphicsOrientationRotated270)
1.539 + {
1.540 + OwnerPanic(EWservPanicInvalidSurfaceConfiguration);
1.541 + }
1.542 +
1.543 + CWsClientWindow* cliWin = CliWin();
1.544 + CScreen* screen = cliWin->Screen();
1.545 + __ASSERT_DEBUG(screen, Panic(EWsPanicNoScreen));
1.546 +
1.547 + CWindowElementSet& set = screen->WindowElements();
1.548 + TBool mustRegister = ETrue;
1.549 + TRect oldExtent(0,0,0,0);
1.550 + MWsElement::TElementRotation oldRotation = MWsElement::EElementAntiClockwise0;
1.551 + TBool oldFlip = EFalse;
1.552 + TRect oldViewport(0,0,0,0);
1.553 + TSurfaceId oldSurfaceId = TSurfaceId::CreateNullId();
1.554 +
1.555 + // If a element has already been set
1.556 + if (HasElement())
1.557 + {
1.558 + TBackgroundAttributes* backgroundAttributes = set.FindBackgroundElement(*cliWin);
1.559 + WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
1.560 +
1.561 + if (backgroundAttributes->iElement)
1.562 + {
1.563 + MWsElement& element = *(backgroundAttributes->iElement);
1.564 + element.GetDestinationRectangle(oldExtent);
1.565 + element.GetSourceRectangle(oldViewport);
1.566 + oldRotation = element.SourceRotation();
1.567 + oldFlip = element.SourceFlipping();
1.568 + oldSurfaceId = element.ConnectedSurface();
1.569 + mustRegister = EFalse;
1.570 + // If it is a different surface, flag to register the new surface
1.571 + if (element.ConnectedSurface() != surfaceId)
1.572 + {
1.573 + mustRegister = ETrue;
1.574 + }
1.575 + }
1.576 + }
1.577 +
1.578 + //the call to AcquireBackgroundElementL() will remove any existing background element
1.579 + TBackgroundAttributes& backgroundAttributes = mustRegister ?
1.580 + AcquireBackgroundElementL() : *(set.FindBackgroundElement(*cliWin));
1.581 + MWsElement& element = *(backgroundAttributes.iElement);
1.582 + TInt err = KErrNone;
1.583 + if (mustRegister)
1.584 + {
1.585 + err = set.RegisterSurface(surfaceId);
1.586 + switch(err)
1.587 + {
1.588 + case KErrBadHandle:
1.589 + // Invalid surface IDs have to return KErrArgument
1.590 + err = KErrArgument;
1.591 + // drop through
1.592 + case KErrNoMemory:
1.593 + case KErrArgument:
1.594 + ReleaseBackgroundElement();
1.595 + User::Leave(err);
1.596 + case KErrNone:
1.597 + break;
1.598 + default:
1.599 + // No need to release layer here since session closure will do it
1.600 + // automatically when the client thread is panicked.
1.601 + OwnerPanic(EWservPanicInvalidSurface);
1.602 + }
1.603 +
1.604 + err = element.ConnectSurface(surfaceId);
1.605 + if (err != KErrNone)
1.606 + {
1.607 + set.UnregisterSurface(surfaceId);
1.608 + ReleaseBackgroundElement(); //Releasing new empty element
1.609 + User::Leave(err);
1.610 + }
1.611 +
1.612 + if (screen->DsaSurface() == surfaceId)
1.613 + {
1.614 + TUint32 flags;
1.615 + element.GetRenderStageFlags(flags);
1.616 + flags |= MWsElement::EElementIsDirectlyRenderedUserInterface;
1.617 + element.SetRenderStageFlags(flags);
1.618 + }
1.619 + }
1.620 +
1.621 + SetHasElement(ETrue); //set element flag
1.622 + SetMayContainElementFlags();
1.623 +
1.624 + err = CWindowElement::SetElement(element,aConfiguration,ETrue); //set viewport and orientation
1.625 + if (err == KErrArgument)
1.626 + {
1.627 + OwnerPanic(EWservPanicInvalidSurfaceConfiguration);
1.628 + }
1.629 + TRect srcRect;
1.630 + aConfiguration.GetViewport(srcRect);
1.631 + if (!srcRect.IsEmpty())
1.632 + backgroundAttributes.SetExplicitViewPort();
1.633 + element.GetSourceRectangle(srcRect);
1.634 + cliWin->SetOriginalSrcElementRect(srcRect);
1.635 +
1.636 + //Set Extent
1.637 + TRect newExtent;
1.638 + aConfiguration.GetExtent(newExtent);
1.639 + SetElementExtentL(newExtent, backgroundAttributes);
1.640 + cliWin->SetOriginalDestElementRect(newExtent);
1.641 +
1.642 + MWsWindowTreeObserver* windowTreeObserver = Screen()->WindowTreeObserver();
1.643 + if (windowTreeObserver && mustRegister)
1.644 + {
1.645 + windowTreeObserver->ElementAdded(*iWsWin, element);
1.646 + }
1.647 +
1.648 + //If set, redraw
1.649 + if (aTriggerRedraw)
1.650 + {
1.651 + TRect newViewport;
1.652 + aConfiguration.GetViewport(newViewport);
1.653 + CFbsBitGc::TGraphicsOrientation orientation = aConfiguration.Orientation();
1.654 + MWsElement::TElementRotation newRotation = GcToElementRotation(orientation);
1.655 + TBool newFlip = aConfiguration.Flip();
1.656 +
1.657 + //The following parameter guarantees that update will be scheduled.
1.658 + //This will trigger the composition.
1.659 + TBool alwaysScheduleUpdate = (oldSurfaceId != surfaceId) ||
1.660 + (oldExtent != newExtent) ||
1.661 + (oldViewport != newViewport) ||
1.662 + (oldRotation != newRotation)||
1.663 + (oldFlip != newFlip);
1.664 +
1.665 + ElementRedraw(oldExtent,newExtent,alwaysScheduleUpdate);
1.666 + }
1.667 + }
1.668 +
1.669 +/**
1.670 +Sets the EMayContainElement flag for parent window.
1.671 +Sets the flag for all ancestor windows.
1.672 +**/
1.673 +void CWsWindowRedraw::SetMayContainElementFlags()
1.674 + {
1.675 + CWsWindowBase* parent = CliWin()->BaseParent();
1.676 + TInt type = parent->WinType();
1.677 + while(type ==EWinTypeClient)
1.678 + {
1.679 + CWsClientWindow* win = static_cast<CWsClientWindow*>(parent);
1.680 + win->Redraw()->iStateFlags |= EMayContainElement;
1.681 + parent=parent->BaseParent();
1.682 + type = parent->WinType();
1.683 + }
1.684 + }
1.685 +
1.686 +void CWsWindowRedraw::SetElementExtentL(TRect& aNewExtent, TBackgroundAttributes& aAttributes)
1.687 + {
1.688 + CWsClientWindow* cliWin = CliWin();
1.689 + MWsElement& element = *(aAttributes.iElement);
1.690 + if (aNewExtent.IsEmpty())
1.691 + {
1.692 + aNewExtent = cliWin->FullRect();
1.693 + aAttributes.SetExplicitExtent(EFalse);
1.694 + }
1.695 + else
1.696 + {
1.697 + TRect tempWindowPosition = cliWin->FullRect(); //get window absolute coordinates
1.698 + aNewExtent.Move(tempWindowPosition.iTl); //shift user defined extent to absolute coordinates
1.699 + aAttributes.SetExplicitExtent(ETrue);
1.700 + }
1.701 + element.SetDestinationRectangle(aNewExtent);
1.702 + }
1.703 +
1.704 +void CWsWindowRedraw::ElementRedraw(const TRect& aOldExtent, const TRect& aNewExtent, TBool aAlwaysScheduleUpdate)
1.705 + {
1.706 + if (!aOldExtent.IsEmpty())
1.707 + {
1.708 + //If the previous extent was different
1.709 + if (aOldExtent != aNewExtent)
1.710 + {
1.711 + STACK_REGION tempRegion;
1.712 + tempRegion.AddRect(aOldExtent);
1.713 + tempRegion.AddRect(aNewExtent);
1.714 +
1.715 + //Calculate the difference between
1.716 + TRect tempRect = aOldExtent;
1.717 + tempRect.Intersection(aNewExtent); //intersect both regions
1.718 + tempRegion.SubRect(tempRect); //cut unaltered region
1.719 + Screen()->ScheduleRegionUpdate(&tempRegion);
1.720 +
1.721 + tempRegion.Close();
1.722 + }
1.723 + else
1.724 + {
1.725 + if(aAlwaysScheduleUpdate)
1.726 + {
1.727 + TTimeIntervalMicroSeconds interval(0);
1.728 + Screen()->ScheduleRender(interval);
1.729 + }
1.730 + }
1.731 + }
1.732 + else
1.733 + {
1.734 + TRegionFix<1> region(aNewExtent);
1.735 + Screen()->ScheduleRegionUpdate(®ion);
1.736 + }
1.737 + }
1.738 +
1.739 +void CWsWindowRedraw::RemoveBackgroundSurface(TBool aTriggerRedraw)
1.740 + {
1.741 + if (HasElement())
1.742 + {
1.743 + TBackgroundAttributes* backgroundAttributes = CliWin()->Screen()->
1.744 + WindowElements().FindBackgroundElement(*CliWin());
1.745 + WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
1.746 + if (backgroundAttributes->iElement)
1.747 + {
1.748 + RemoveBackgroundElement(aTriggerRedraw);
1.749 + }
1.750 + }
1.751 + }
1.752 +
1.753 +void CWsWindowRedraw::RemoveBackgroundElement(TBool aTriggerRedraw)
1.754 + {
1.755 + CWsClientWindow* cliWin = CliWin();
1.756 + CScreen* screen = cliWin->Screen();
1.757 + TRect tempRect;
1.758 + if (aTriggerRedraw)
1.759 + {
1.760 + TBackgroundAttributes* backgroundAttributes = screen->WindowElements().FindBackgroundElement(*CliWin());
1.761 + WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
1.762 + if (backgroundAttributes->ExplicitExtent())
1.763 + {
1.764 + backgroundAttributes->iElement->GetDestinationRectangle(tempRect);
1.765 + backgroundAttributes->SetExplicitExtent(EFalse);
1.766 + }
1.767 + else
1.768 + {
1.769 + tempRect = cliWin->FullRect();
1.770 + }
1.771 + }
1.772 + ReleaseBackgroundElement();
1.773 + if (aTriggerRedraw)
1.774 + {
1.775 + if (screen->ChangeTracking())
1.776 + {
1.777 + TTimeIntervalMicroSeconds interval(0);
1.778 + screen->ScheduleRender(interval);
1.779 + }
1.780 + else
1.781 + {
1.782 + TRegionFix<1> region(tempRect);
1.783 + screen->ScheduleRegionUpdate(®ion);
1.784 + }
1.785 + }
1.786 + }
1.787 +
1.788 +void CWsWindowRedraw::GetBackgroundSurfaceL(TSurfaceConfiguration& aConfiguration)
1.789 + {
1.790 + if (aConfiguration.Size() < sizeof(TSurfaceConfiguration))
1.791 + {
1.792 + __ASSERT_COMPILE(sizeof(TSurfaceConfiguration2)==sizeof(TSurfaceConfiguration));
1.793 + if (aConfiguration.Size() != sizeof(TSurfaceConfiguration1))
1.794 + {
1.795 + OwnerPanic(EWservPanicInvalidSurfaceConfiguration);
1.796 + }
1.797 + }
1.798 +
1.799 + CWsClientWindow* cliWin = CliWin();
1.800 + TBackgroundAttributes* backgroundAttributes = NULL;
1.801 +
1.802 + if (HasElement())
1.803 + {
1.804 + backgroundAttributes = cliWin->Screen()->WindowElements().FindBackgroundElement(*cliWin);
1.805 + WS_ASSERT_DEBUG(backgroundAttributes,EWsPanicNoWindowElement);
1.806 + if (!backgroundAttributes->iElement)
1.807 + {
1.808 + User::Leave(KErrNotFound);
1.809 + }
1.810 + }
1.811 + else
1.812 + {
1.813 + User::Leave(KErrNotFound);
1.814 + }
1.815 +
1.816 + MWsElement& element = *(backgroundAttributes->iElement);
1.817 +
1.818 + TInt errCode=CWindowElementSet::GetConfiguration(aConfiguration,element);
1.819 +
1.820 + // Get source rect
1.821 + if (errCode>=KErrNone)
1.822 + {
1.823 + if (!backgroundAttributes->ExplicitViewPort())
1.824 + {
1.825 + aConfiguration.SetViewport(TRect());
1.826 + }
1.827 + else
1.828 + {
1.829 + TRect tempExtent = cliWin->GetOriginalSrcElementRect();
1.830 + aConfiguration.SetViewport(tempExtent);
1.831 + }
1.832 + }
1.833 +
1.834 + //Convert and copy extent
1.835 + if (errCode>=KErrNone)
1.836 + {
1.837 + if (!backgroundAttributes->ExplicitExtent())
1.838 + {
1.839 + aConfiguration.SetExtent(TRect());
1.840 + }
1.841 + else //translate to window coordinates
1.842 + {
1.843 + TRect tempExtent = cliWin->GetOriginalDestElementRect();
1.844 + tempExtent.Move(-cliWin->Origin());
1.845 + aConfiguration.SetExtent(tempExtent);
1.846 + }
1.847 + }
1.848 + }
1.849 +
1.850 +void CWsWindowRedraw::ReleaseBackgroundElement()
1.851 + {
1.852 + if (HasElement())
1.853 + {
1.854 + CWsClientWindow* cliWin = CliWin();
1.855 + CScreen* screen = cliWin->Screen();
1.856 + screen->WindowElements().ReleaseBackgroundElement(*cliWin, ETrue);
1.857 + screen->ElementRemoved();
1.858 + }
1.859 + }
1.860 +
1.861 +//
1.862 +// Blank window //
1.863 +//
1.864 +
1.865 +CWsBlankWindow::CWsBlankWindow(CWsWindow *aWin) : CWsWindowRedraw(aWin), iColor(iWsWin->RootWindow()->DefaultBackgroundColor()), iNoColor(EFalse)
1.866 + {
1.867 + }
1.868 +
1.869 +CWsBlankWindow::~CWsBlankWindow()
1.870 + {
1.871 + }
1.872 +
1.873 +void CWsBlankWindow::ConstructL()
1.874 + {
1.875 + CWsWindowRedraw::ConstructL();
1.876 + if (Screen()->ChangeTracking())
1.877 + {
1.878 + STACK_REGION dirtyRegion;
1.879 + dirtyRegion.Copy(iWsWin->WindowArea());
1.880 + dirtyRegion.Offset(-iWsWin->Origin());
1.881 + iWsWin->AddDirtyWindowRegion(dirtyRegion);
1.882 + dirtyRegion.Close();
1.883 + }
1.884 + }
1.885 +
1.886 +void CWsBlankWindow::SetColor(TRgb aColor)
1.887 + {
1.888 + iColor=aColor;
1.889 + iNoColor=EFalse;
1.890 + if (Screen()->ChangeTracking())
1.891 + {
1.892 + STACK_REGION dirtyRegion;
1.893 + dirtyRegion.Copy(iWsWin->WindowArea());
1.894 + dirtyRegion.Offset(-iWsWin->Origin());
1.895 + iWsWin->AddDirtyWindowRegion(dirtyRegion);
1.896 + dirtyRegion.Close();
1.897 +
1.898 + if (iWsWin->IsActive() && iWsWin->IsVisible())
1.899 + {
1.900 + Screen()->ScheduleWindow(iWsWin);
1.901 + }
1.902 + }
1.903 + else
1.904 + {
1.905 + Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
1.906 + }
1.907 + }
1.908 +
1.909 +TBool CWsBlankWindow::CommandL(TInt aOpcode, TWsWinCmdUnion &aCmd)
1.910 + {
1.911 + switch(aOpcode)
1.912 + {
1.913 + case EWsWinOpSetBackgroundSurface:
1.914 + SetBackgroundSurfaceL(*aCmd.Surface);
1.915 + break;
1.916 + case EWsWinOpSetBackgroundSurfaceConfig:
1.917 + SetBackgroundSurfaceL(aCmd.SurfaceConfigurationAndTrigger->surfaceConfig, aCmd.SurfaceConfigurationAndTrigger->triggerRedraw, EFalse);
1.918 + break;
1.919 + case EWsWinOpRemoveBackgroundSurface:
1.920 + RemoveBackgroundSurface(*aCmd.Bool);
1.921 + break;
1.922 + case EWsWinOpGetBackgroundSurfaceConfig:
1.923 + {
1.924 + TSurfaceConfiguration tempConfiguration = *aCmd.SurfaceConfiguration;
1.925 + GetBackgroundSurfaceL(tempConfiguration);
1.926 + TInt tempSize = aCmd.SurfaceConfiguration->Size();
1.927 + if (sizeof(TSurfaceConfiguration)<tempSize)
1.928 + tempSize = sizeof(TSurfaceConfiguration);
1.929 + CWsClient::ReplyBuf(&tempConfiguration,tempSize);
1.930 + }
1.931 + break;
1.932 + case EWsWinOpSetColor:
1.933 + SetColor(*aCmd.rgb);
1.934 + break;
1.935 + case EWsWinOpSetNoBackgroundColor:
1.936 + SetBackgroundClear();
1.937 + break;
1.938 + default:
1.939 + return(EFalse);
1.940 + }
1.941 + return(ETrue);
1.942 + }
1.943 +
1.944 +TRgb CWsBlankWindow::BackColor() const
1.945 + {
1.946 + return(iColor);
1.947 + }
1.948 +
1.949 +TBool CWsBlankWindow::GetRedrawRect(TRect &) const
1.950 + {
1.951 + if (!iNoColor || iWsWin->iAnimList)
1.952 + iWsWin->Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
1.953 + return(EFalse);
1.954 + }
1.955 +
1.956 +TBool CWsBlankWindow::NeedsRedraw() const
1.957 + {
1.958 + return(EFalse);
1.959 + }
1.960 +
1.961 +void CWsBlankWindow::DrawWindow()
1.962 + {
1.963 + if ((!iNoColor)||HasElement())
1.964 + {
1.965 + DrawBackgroundColor(*iRedrawRegion,!iNoColor);
1.966 + }
1.967 + }