1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nonnga/SERVER/WINDOW.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,535 @@
1.4 +// Copyright (c) 1994-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +// Window functions
1.18 +//
1.19 +//
1.20 +
1.21 +#include "server.h"
1.22 +#include "rootwin.h"
1.23 +#include "window.h"
1.24 +#include "walkwindowtree.h"
1.25 +#include "wstop.h"
1.26 +#include "ScrDev.H"
1.27 +#include "EVENT.H"
1.28 +#include "ANIM.H"
1.29 +#include "Direct.H"
1.30 +#include "panics.h"
1.31 +#include "backedupwindow.h"
1.32 +#include "wstypes.h"
1.33 +#include "offscreenbitmap.h"
1.34 +
1.35 +CWsWindow* CWsWindow::iAccessListRecentEnd = 0;
1.36 +CWsWindow* CWsWindow::iAccessListOldestEnd = 0;
1.37 +
1.38 +CWsWindow::CWsWindow(CWsClient* aOwner,WH_HANDLES aType, CScreen* aScreen) : CWsWindowBase(aOwner,aType,aScreen),
1.39 + iFlags(EFlagHidden),
1.40 + iDSAs(_FOFF(CWsDirectScreenAccess,iMultipleDSALink))
1.41 + {
1.42 + }
1.43 +
1.44 +CWsWindow::~CWsWindow()
1.45 + {
1.46 + iFadableRegion.Close();
1.47 + }
1.48 +
1.49 +void CWsWindow::Construct()
1.50 + {
1.51 + InsertIntoAccessListOldestEnd();
1.52 + }
1.53 +
1.54 +void CWsWindow::Shutdown()
1.55 + {
1.56 + TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EWindowClosing, 0, 0, this));
1.57 + RemoveFromAccessList();
1.58 + CWsAnim::WindowClosing(iAnimList); // Destroy any animated objects attached to this window
1.59 + iVisibleRegion.Close();
1.60 + iScheduledRegion.Close();
1.61 + AbortAllDSAs();
1.62 + CWsWindowBase::Shutdown();
1.63 + SetPointerCursor(NULL);
1.64 + delete iRedraw;
1.65 + }
1.66 +
1.67 +//
1.68 +// Region and invalid area functions //
1.69 +//
1.70 +
1.71 +TRect CWsWindow::FullRect() const
1.72 + {
1.73 + return(TRect(iOrigin,iRel.Size()));
1.74 + }
1.75 +
1.76 +//
1.77 +// Normal regions //
1.78 +//
1.79 +
1.80 +void CWsWindow::SetNonFading(TBool aNonFade)
1.81 + {
1.82 + if (aNonFade)
1.83 + iFlags|=EFlagNonFadingWindow;
1.84 + else
1.85 + iFlags&=(~EFlagNonFadingWindow);
1.86 + }
1.87 +
1.88 +TPoint CWsWindow::InquireOffset(TUint32 aHandle) const
1.89 + {
1.90 + CWsWindowBase *win2;
1.91 + iWsOwner->HandleToWindow(aHandle,&win2);
1.92 + return(iOrigin-win2->Origin());
1.93 + }
1.94 +
1.95 +TDisplayMode CWsWindow::DisplayMode() const
1.96 + {
1.97 + return iScreen->DisplayMode();
1.98 + }
1.99 +
1.100 +void CWsWindow::StatusDump(TDes &aBuf)
1.101 + {
1.102 + _LIT(KStatusString1,"CWsWindow[0x%x]RWindow[0x%x,%d],Pos(%d,%d),Size(%d,%d)");
1.103 + _LIT(KStatusString3,",Mode=%d");
1.104 + _LIT(KStatusInvisible,",Inv");
1.105 + aBuf.AppendFormat(KStatusString1,this,iClientHandle,LogHandle(),iRel.iTl.iX,iRel.iTl.iY,Size().iWidth,Size().iHeight);
1.106 + aBuf.AppendFormat(KStatusString3,(TInt)iScreen->DisplayMode());
1.107 + if (!IsVisible())
1.108 + aBuf.Append(KStatusInvisible);
1.109 + }
1.110 +
1.111 +TDblQue<TPointerKeyList> *CWsWindow::PointerKeyList() const
1.112 + {
1.113 + return(NULL);
1.114 + }
1.115 +
1.116 +void CWsWindow::AbortAllDSAs()
1.117 + {
1.118 + iScreen->AbortDSAs(RDirectScreenAccess::ETerminateRegion,iDSAs);
1.119 + }
1.120 +
1.121 +void CWsWindow::PossibleVisibilityChangedEvent(TBool aForceSend)
1.122 + {
1.123 + // notify plugin
1.124 + //
1.125 + TWservCrEvent crEvent(TWservCrEvent::EWindowVisibilityChanged, HasBeenDrawnToScreen(), IsVisible()? &iVisibleRegion : NULL, this);
1.126 + TWindowServerEvent::NotifyDrawer(crEvent);
1.127 +
1.128 + if (!(iFlags & EFlagGeneratesVisibilityEvents))
1.129 + return;
1.130 +
1.131 + if (!IsVisible())
1.132 + {
1.133 + goto notvisible;
1.134 + }
1.135 +
1.136 + if (iVisibleRegion.Count() == 0)
1.137 + {
1.138 + goto notvisible;
1.139 + }
1.140 +
1.141 + {// braces here to avoid gccxml error
1.142 + TInt visibleArea = 0;
1.143 + TInt count = iVisibleRegion.Count();
1.144 + TInt ii;
1.145 + for (ii = 0; ii < count; ii++)
1.146 + {
1.147 + visibleArea+= iVisibleRegion[ii].Width() * iVisibleRegion[ii].Height();
1.148 + }
1.149 +
1.150 + const TRegion* baseRegion = (static_cast<CWsClientWindow*>(this))->BaseArea();
1.151 + TInt baseArea = 0;
1.152 + count = baseRegion->Count();
1.153 + for (ii = 0; ii < count; ii++)
1.154 + {
1.155 + const TRect& rect = (*baseRegion)[ii];
1.156 + baseArea+= rect.Width() * rect.Height();
1.157 + }
1.158 +
1.159 + if (visibleArea == baseArea)
1.160 + {
1.161 + goto fullyvisible;
1.162 + }
1.163 + else
1.164 + {
1.165 + goto partiallyvisible;
1.166 + }
1.167 + }
1.168 +
1.169 +fullyvisible:
1.170 + if (aForceSend || !(iFlags & EFlagNotCantBeSeen) || !(iFlags & EFlagCanBeSeen))
1.171 + {
1.172 + iFlags |= (EFlagCanBeSeen | EFlagNotCantBeSeen);
1.173 + TWindowServerEvent::SendVisibilityChangedEvents(this, TWsVisibilityChangedEvent::EPartiallyVisible | TWsVisibilityChangedEvent::EFullyVisible);
1.174 + // we have to set EPartiallyVisible too for compatibility reasons
1.175 + }
1.176 + return;
1.177 +
1.178 +partiallyvisible:
1.179 + if (aForceSend || !(iFlags & EFlagCanBeSeen) || (iFlags & EFlagNotCantBeSeen))
1.180 + {
1.181 + iFlags |= EFlagCanBeSeen;
1.182 + iFlags &= ~EFlagNotCantBeSeen;
1.183 + TWindowServerEvent::SendVisibilityChangedEvents(this, TWsVisibilityChangedEvent::EPartiallyVisible);
1.184 + }
1.185 + return;
1.186 +
1.187 +notvisible:
1.188 + if (aForceSend || (iFlags & EFlagCanBeSeen) || (iFlags & EFlagNotCantBeSeen))
1.189 + {
1.190 + iFlags &= ~(EFlagCanBeSeen | EFlagNotCantBeSeen);
1.191 + TWindowServerEvent::SendVisibilityChangedEvents(this, TWsVisibilityChangedEvent::ENotVisible);
1.192 + }
1.193 + return;
1.194 + }
1.195 +
1.196 +TPoint CWsWindow::Origin() const
1.197 + {
1.198 + return iOrigin;
1.199 + }
1.200 +
1.201 +TRect CWsWindow::AbsRect() const
1.202 + {
1.203 + return iAbs;
1.204 + }
1.205 +
1.206 +TSize CWsWindow::Size() const
1.207 + {
1.208 + return iRel.Size();
1.209 + }
1.210 +
1.211 +TBool CWsWindow::SetScheduledRegion(const RWsRegion& aRegion)
1.212 + {
1.213 + iScheduledRegion.Copy(aRegion);
1.214 + return !iScheduledRegion.CheckError();
1.215 + }
1.216 +
1.217 +void CWsWindow::ClearScheduledRegion()
1.218 + {
1.219 + iScheduledRegion.Clear();
1.220 + }
1.221 +
1.222 +void CWsWindow::SetFadeBehind(TBool aFade)
1.223 + {
1.224 + if (aFade != ((iFlags & EFlagFadeBehind) != 0))
1.225 + {
1.226 + iFlags ^= EFlagFadeBehind;
1.227 + }
1.228 + }
1.229 +
1.230 +void CWsWindow::FocusChanged(TBool aNewFocusState)
1.231 + {
1.232 + TRAPD(err,FocusChangedL(aNewFocusState));
1.233 + if (err!=KErrNone)
1.234 + OwnerPanic(EWservPanicAnimLeave);
1.235 + }
1.236 +
1.237 +void CWsWindow::FocusChangedL(TBool aNewFocusState)
1.238 + {
1.239 + for (CWsAnim * anim = iAnimList; anim; anim = anim->Next())
1.240 + {
1.241 + anim->FocusChanged(aNewFocusState);
1.242 + }
1.243 + }
1.244 +
1.245 +// Moves a window to the recent end of the accessed list
1.246 +void CWsWindow::Accessed()
1.247 + {
1.248 + WS_ASSERT_DEBUG(iAccessListRecentEnd && iAccessListOldestEnd, EWsPanicAccessList);
1.249 +
1.250 + RemoveFromAccessList();
1.251 + InsertIntoAccessListRecentEnd();
1.252 + }
1.253 +
1.254 +void CWsWindow::InsertIntoAccessListOldestEnd()
1.255 + {
1.256 + iAccessListPrev = 0;
1.257 + iAccessListNext = iAccessListOldestEnd;
1.258 + if (iAccessListNext)
1.259 + iAccessListNext->iAccessListPrev = this;
1.260 + iAccessListOldestEnd = this;
1.261 + if (!iAccessListRecentEnd)
1.262 + iAccessListRecentEnd = this;
1.263 + }
1.264 +
1.265 +void CWsWindow::InsertIntoAccessListRecentEnd()
1.266 + {
1.267 + iAccessListNext = 0;
1.268 + iAccessListPrev = iAccessListRecentEnd;
1.269 + if (iAccessListPrev)
1.270 + iAccessListPrev->iAccessListNext = this;
1.271 + iAccessListRecentEnd = this;
1.272 + if (!iAccessListOldestEnd)
1.273 + iAccessListOldestEnd = this;
1.274 + }
1.275 +
1.276 +void CWsWindow::RemoveFromAccessList()
1.277 + {
1.278 + if (iAccessListOldestEnd == this)
1.279 + iAccessListOldestEnd = iAccessListNext;
1.280 + if (iAccessListRecentEnd == this)
1.281 + iAccessListRecentEnd = iAccessListPrev;
1.282 + if (iAccessListPrev)
1.283 + iAccessListPrev->iAccessListNext = iAccessListNext;
1.284 + if (iAccessListNext)
1.285 + iAccessListNext->iAccessListPrev = iAccessListPrev;
1.286 + }
1.287 +
1.288 +TBool CWsWindow::ReleaseMemory(MWsMemoryRelease::TMemoryReleaseLevel aLevel)
1.289 + {
1.290 + for (CWsWindow * access = iAccessListOldestEnd; access; access = access->iAccessListNext)
1.291 + {
1.292 + if(access->Redraw())
1.293 + {
1.294 + if (access->Redraw()->ReleaseMemory(aLevel))
1.295 + return ETrue;
1.296 + }
1.297 + }
1.298 + return EFalse;
1.299 + }
1.300 +
1.301 +const TRegion& CWsWindow::VisibleRegion() const
1.302 + {
1.303 + return(iVisibleRegion);
1.304 + }
1.305 +
1.306 +TBool CWsWindow::IsDSAHost() const
1.307 + {
1.308 + TBool res = EFalse;
1.309 + if ( !iDSAs.IsEmpty() )
1.310 + {
1.311 + TSglQueIter<CWsDirectScreenAccess> iter( (TSglQueBase&)iDSAs );
1.312 + iter.SetToFirst();
1.313 + CWsDirectScreenAccess* dsa;
1.314 + while ( (dsa = iter++) != NULL && !res )
1.315 + {
1.316 + res = dsa->IsVisible() || dsa->IsSyncTimeoutPending();
1.317 + }
1.318 + }
1.319 + return res;
1.320 + }
1.321 +
1.322 +void CWsWindow::ResetVisibleRegion()
1.323 + {
1.324 + if (!iVisibleRegion.IsEmpty())
1.325 + {
1.326 + iVisibleRegion.Clear();
1.327 + }
1.328 + iFadableRegion.Clear();
1.329 + }
1.330 +
1.331 +void CWsWindow::SetVisibleRegion(const TRegion& aNewRegion, const TRegion* aTop, TRegion& aNewFadableRegion)
1.332 + {
1.333 + STACK_REGION difference;
1.334 + TBool diffs = EFalse;
1.335 +
1.336 + difference.Copy(iVisibleRegion);
1.337 + difference.SubRegion(aNewRegion);
1.338 + if (!difference.IsEmpty())
1.339 + {
1.340 + diffs = ETrue;
1.341 + if (IsTranslucent())
1.342 + {
1.343 + // Andy - If this is a client window (what else could it be) we can also subtract the
1.344 + // user defined opaque region before doing this:
1.345 + iScreen->AddRedrawRegion(difference, EFalse);
1.346 + }
1.347 + }
1.348 +
1.349 + difference.Copy(aNewRegion);
1.350 + if (HasBeenDrawnToScreen())
1.351 + {
1.352 + difference.SubRegion(iVisibleRegion);
1.353 + }
1.354 + if (!difference.IsEmpty())
1.355 + {
1.356 + diffs = ETrue;
1.357 + STACK_REGION topDiff;
1.358 + topDiff.Copy(difference);
1.359 + WS_ASSERT_DEBUG(aTop,EWsPanicRegion);
1.360 + topDiff.Intersect(*aTop);
1.361 + difference.SubRegion(topDiff);
1.362 + iScreen->AddRedrawRegion(topDiff, EFalse, ERedrawTopOnly);
1.363 + iScreen->AddRedrawRegion(difference, EFalse, ERedrawAll);
1.364 + topDiff.Close();
1.365 + }
1.366 +
1.367 + difference.Close();
1.368 +
1.369 + AbortDsaIfRequired(aNewRegion, aTop);
1.370 +
1.371 + if (diffs)
1.372 + {
1.373 + ResetVisibleRegion();
1.374 + iVisibleRegion.Copy(aNewRegion);
1.375 + PossibleVisibilityChangedEvent(EFalse);
1.376 + }
1.377 +
1.378 + iFadableRegion.Copy( aNewFadableRegion );
1.379 +
1.380 + // Just because the visible region (screen coordinates) didn't change doesn't
1.381 + // mean the invalid region (window coordinates) didn't change, so we always call this.
1.382 + iRedraw->VisibleRegionChange();
1.383 + }
1.384 +
1.385 +void CWsWindow::ClearVisibleRegion()
1.386 + {
1.387 + AbortAllDSAs();
1.388 + iScreen->AddRedrawRegion(VisibleRegion(), EFalse);
1.389 + ResetVisibleRegion();
1.390 + PossibleVisibilityChangedEvent(EFalse);
1.391 + iFlags &= ~EFlagDrawnToScreen;
1.392 + }
1.393 +
1.394 +void CWsWindow::AbortDsaIfRequired(const TRegion& aNewRegion, const TRegion* aTop)
1.395 + {
1.396 + if (!iDSAs.IsEmpty())
1.397 + {
1.398 + // If the top visible region of this window has changed, DSA clients may need
1.399 + // to be sent a DSA abort, as they may be drawing to a different region
1.400 + STACK_REGION newTopVisible;
1.401 + newTopVisible.Copy(aNewRegion); // new visible region
1.402 + if (aTop!=NULL)
1.403 + {
1.404 + newTopVisible.Intersect(*aTop); // area of new visible region not obscured by any other opaque or translucent windows
1.405 + }
1.406 + // Build a list of DSA clients that need to be sent a DSA abort
1.407 + TSglQue<CWsDirectScreenAccess> dsaList(_FOFF(CWsDirectScreenAccess,iAbortLink));
1.408 + TSglQueIter<CWsDirectScreenAccess> iter(iDSAs);
1.409 + CWsDirectScreenAccess* dsa;
1.410 + while ((dsa=iter++)!=NULL)
1.411 + {
1.412 + if (dsa->IsAbortRequired(newTopVisible))
1.413 + {
1.414 + dsaList.AddLast(*dsa);
1.415 + }
1.416 + }
1.417 + if (!dsaList.IsEmpty())
1.418 + {
1.419 + iScreen->AbortDSAs(RDirectScreenAccess::ETerminateRegion, dsaList);
1.420 + }
1.421 + newTopVisible.Close();
1.422 + }
1.423 + }
1.424 +
1.425 +const TRegion* CWsWindow::VisibleRegionIfValid() const
1.426 + {
1.427 + return iVisibleRegion.CheckError() ? NULL : &iVisibleRegion;
1.428 + }
1.429 +
1.430 +TBool CWsWindow::ReadyToDraw() const
1.431 + {
1.432 + return iRedraw->ReadyToDraw();
1.433 + }
1.434 +
1.435 +/**
1.436 +This function draws the region specified
1.437 +Andy - I would be happier if no region was specified here - the GC should incorporate that information
1.438 +*/
1.439 +void CWsWindow::Render(CFbsBitGc * aGc, const TRegion& aRegion)
1.440 + {
1.441 + WS_ASSERT_DEBUG(IsVisible() || (WinType() == EWinTypeRoot), EWsPanicScheduledRedraw);
1.442 +
1.443 + Accessed();
1.444 + iFlags |= EFlagDrawnToScreen;
1.445 +
1.446 + iRedraw->PreDrawWindow(aGc, aRegion);
1.447 + iRedraw->DrawWindow();
1.448 + iRedraw->PostDrawWindow(aGc);
1.449 + }
1.450 +
1.451 +void CWsWindow::SetNextScheduled(CWsWindow * aWin)
1.452 + {
1.453 + iNextScheduled = aWin;
1.454 + }
1.455 +
1.456 +CWsWindow * CWsWindow::NextScheduled() const
1.457 + {
1.458 + return iNextScheduled;
1.459 + }
1.460 +
1.461 +void CWsWindow::AddSprite(CWsSpriteBase * aSprite)
1.462 + {
1.463 + aSprite->SetNext(iSpriteList);
1.464 + iSpriteList = aSprite;
1.465 + }
1.466 +
1.467 +void CWsWindow::RemoveSprite(CWsSpriteBase * aSprite)
1.468 + {
1.469 + if (aSprite == iSpriteList)
1.470 + {
1.471 + iSpriteList = aSprite->Next();
1.472 + }
1.473 + else
1.474 + {
1.475 + for (CWsSpriteBase * sprite = iSpriteList; sprite; sprite = sprite->Next())
1.476 + {
1.477 + if (sprite->Next() == aSprite)
1.478 + {
1.479 + sprite->SetNext(aSprite->Next());
1.480 + }
1.481 + }
1.482 + }
1.483 + aSprite->SetNext(0);
1.484 + }
1.485 +
1.486 +void CWsWindow::DeactivateAllSprites()
1.487 + {
1.488 + CWsSpriteBase * current = iSpriteList;
1.489 + while (current)
1.490 + {
1.491 + CWsSpriteBase * next = current->Next();
1.492 + current->Deactivate();
1.493 + current = next;
1.494 + }
1.495 + }
1.496 +
1.497 +void CWsWindow::ClipRectToViewport(TRect& aRect) const
1.498 + {
1.499 + const CWsWindowBase * win = this;
1.500 + while (win)
1.501 + {
1.502 + if (win->WinType() != EWinTypeGroup)
1.503 + {
1.504 + aRect.Intersection(win->AbsRect());
1.505 + }
1.506 + win = win->BaseParent();
1.507 + }
1.508 + }
1.509 +
1.510 +const TRegion& CWsWindow::FadableRegion() const
1.511 + {
1.512 + return iFadableRegion;
1.513 + }
1.514 +
1.515 +/**
1.516 +MWsWindow
1.517 +*/
1.518 +MWsWindow * CWsWindow::FindChildByHandle(TUint32 aHandle)
1.519 + {
1.520 + TWalkWindowTreeFindByHandle wwt(aHandle);
1.521 + WalkWindowTree(wwt, EWalkChildren);
1.522 + return wwt.Found();
1.523 + }
1.524 +
1.525 +TUint32 CWsWindow::Handle() const
1.526 + {
1.527 + return ClientHandle();
1.528 + }
1.529 +
1.530 +MWsScreen * CWsWindow::WsScreen() const
1.531 + {
1.532 + return iScreen;
1.533 + }
1.534 +
1.535 +TInt CWsWindow::OrdinalPriority() const
1.536 + {
1.537 + return iOrdinalPriority;
1.538 + }