1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nga/SERVER/openwfc/SPRITE.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,1167 @@
1.4 +// Copyright (c) 1999-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 +// Sprite
1.18 +//
1.19 +//
1.20 +
1.21 +#include <e32std.h>
1.22 +#include "sprite.h"
1.23 +#include "rootwin.h"
1.24 +#include "windowgroup.h"
1.25 +#include "ScrDev.H"
1.26 +#include "wstop.h"
1.27 +#include "ANIM.H"
1.28 +#include "panics.h"
1.29 +#include "EVENT.H"
1.30 +#include "bitgditomwsgraphicscontextmappings.h"
1.31 +#include "../debuglog/DEBUGLOG.H"
1.32 +
1.33 +static const TInt64 KFlashHalfSecond(500000); //interval for flashing sprites
1.34 +
1.35 +GLDEF_D CWsDeltaTimer *CWsSpriteBase::iDeltaTimer=NULL;
1.36 +
1.37 +TInt TimerCallBack(TAny *aPtr)
1.38 + {
1.39 + ((CWsSpriteBase *)aPtr)->TimerExpired();
1.40 + return(KErrNone);
1.41 + }
1.42 +
1.43 +#ifndef _DEBUG
1.44 +
1.45 +#define LOG_SPRITE_REDRAW_START(sprite)
1.46 +#define LOG_SPRITE_REDRAW_END(sprite)
1.47 +#define LOG_SPRITE_FLASH(aSprite)
1.48 +
1.49 +#else
1.50 +
1.51 +#define LOG_SPRITE_REDRAW_START(sprite) LogSpriteRedrawStart(sprite)
1.52 +#define LOG_SPRITE_REDRAW_END(sprite) LogSpriteRedrawEnd(sprite)
1.53 +#define LOG_SPRITE_FLASH(sprite) LogSpriteFlash(sprite)
1.54 +
1.55 +extern CDebugLogBase *wsDebugLog;
1.56 +
1.57 +
1.58 +static void LogSpriteRedrawStart(const CWsSpriteBase& aSprite)
1.59 + {
1.60 + if (wsDebugLog)
1.61 + {
1.62 + _LIT(KAnnotateSpriteRedrawStart, ">> MWsDrawAnnotationObserver::FloatingSpriteRedrawStart [%S][app %d]");
1.63 + const TDesC& clientName = aSprite.WsOwner()->Client().FullName();
1.64 + TBuf<LogTBufSize> log;
1.65 + TTruncateOverflow overflow;
1.66 + log.AppendFormat(KAnnotateSpriteRedrawStart, &overflow, &clientName, aSprite.WsOwner()->ConnectionHandle());
1.67 + wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.68 + }
1.69 + }
1.70 +
1.71 +static void LogSpriteRedrawEnd(const CWsSpriteBase& aSprite)
1.72 + {
1.73 + if (wsDebugLog)
1.74 + {
1.75 + _LIT(KAnnotateSpriteRedrawEnd, "<< MWsDrawAnnotationObserver::FloatingSpriteRedrawEnd [%S][app %d]");
1.76 + const TDesC& clientName = aSprite.WsOwner()->Client().FullName();
1.77 + TBuf<LogTBufSize> log;
1.78 + TTruncateOverflow overflow;
1.79 + log.AppendFormat(KAnnotateSpriteRedrawEnd, &overflow, &clientName, aSprite.WsOwner()->ConnectionHandle());
1.80 + wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.81 + }
1.82 + }
1.83 +
1.84 +static void LogSpriteFlash(const CWsSpriteBase& /*aSprite*/)
1.85 + {
1.86 + if (wsDebugLog)
1.87 + {
1.88 + // The following code causes Exception and is commented out, see defect GFX09962
1.89 +
1.90 +// _LIT(KAnnotateSpriteFlash, "-- MWsDrawAnnotationObserver::SpriteFlash:%d [%S][app %d]");
1.91 +// const TDesC& clientName = aSprite.WsOwner()->Client().FullName();
1.92 +// TBuf<LogTBufSize> log;
1.93 +// TTruncateOverflow overflow;
1.94 +// log.AppendFormat(KAnnotateSpriteFlash, &overflow, &clientName, aSprite.WsOwner()->ConnectionHandle());
1.95 +// wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.96 +
1.97 + // This code is temporarily here until above problem is resolved
1.98 + _LIT(KAnnotateSpriteFlash, "-- MWsDrawAnnotationObserver::SpriteFlash");
1.99 + TBuf<LogTBufSize> log;
1.100 + log.AppendFormat(KAnnotateSpriteFlash);
1.101 + wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.102 + }
1.103 + }
1.104 +
1.105 +
1.106 +#endif
1.107 +
1.108 +
1.109 +
1.110 +static void AnnotateSpriteRedrawStart(const CWsSpriteBase& aSprite, const TRegion& aRegion)
1.111 + {
1.112 + LOG_SPRITE_REDRAW_START(aSprite);
1.113 + MWsDrawAnnotationObserver* annoObs = aSprite.Screen()->DrawAnnotationObserver();
1.114 + if(annoObs)
1.115 + {
1.116 + annoObs->SpriteRedrawStart(aSprite, aRegion);
1.117 + }
1.118 + }
1.119 +
1.120 +static void AnnotateSpriteRedrawEnd(const CWsSpriteBase& aSprite)
1.121 + {
1.122 + LOG_SPRITE_REDRAW_END(aSprite);
1.123 + MWsDrawAnnotationObserver* annoObs = aSprite.Screen()->DrawAnnotationObserver();
1.124 + if(annoObs)
1.125 + {
1.126 + annoObs->SpriteRedrawEnd(aSprite);
1.127 + }
1.128 + }
1.129 +
1.130 +static void AnnotateSpriteFlash(const CWsSpriteBase& aSprite, TBool aFlashOn)
1.131 + {
1.132 + LOG_SPRITE_FLASH(aSprite);
1.133 + MWsDrawAnnotationObserver* annoObs = aSprite.Screen()->DrawAnnotationObserver();
1.134 + if(annoObs)
1.135 + {
1.136 + annoObs->SpriteFlash(aSprite, aFlashOn);
1.137 + }
1.138 + }
1.139 +
1.140 +
1.141 +//
1.142 +// CWsSpriteMember
1.143 +//
1.144 +
1.145 +CWsSpriteMember::CWsSpriteMember()
1.146 + {
1.147 + }
1.148 +
1.149 +CWsSpriteMember::~CWsSpriteMember()
1.150 + {
1.151 + delete iBitmap;
1.152 + delete iMaskBitmap;
1.153 + }
1.154 +
1.155 +TBool CWsSpriteMember::SetL(const TCmdSpriteMember &aCmdSpriteMember)
1.156 + {
1.157 + if (aCmdSpriteMember.iBitmap!=0) // Null member of sequence, time is only valid field in member data
1.158 + {
1.159 + iBitmap=new(ELeave) CFbsBitmap();
1.160 + TInt ret = iBitmap->Duplicate(aCmdSpriteMember.iBitmap);
1.161 + if (ret == KErrNoMemory)
1.162 + {
1.163 + User::Leave(ret);
1.164 + }
1.165 + if (ret != KErrNone)
1.166 + return(ETrue);
1.167 +
1.168 + if (aCmdSpriteMember.iMaskBitmap)
1.169 + {
1.170 + iMaskBitmap=new(ELeave) CFbsBitmap();
1.171 + TInt ret = iMaskBitmap->Duplicate(aCmdSpriteMember.iMaskBitmap);
1.172 + if (ret == KErrNoMemory)
1.173 + {
1.174 + User::Leave(ret);
1.175 + }
1.176 + if (ret != KErrNone)
1.177 + return(ETrue);
1.178 + }
1.179 + iInvertMask=aCmdSpriteMember.iInvertMask;
1.180 + iDrawMode=aCmdSpriteMember.iDrawMode;
1.181 + iOffset=aCmdSpriteMember.iOffset;
1.182 + }
1.183 + iInterval=aCmdSpriteMember.iInterval;
1.184 + return(EFalse);
1.185 + }
1.186 +
1.187 +//
1.188 +// CWsSpriteBase
1.189 +//
1.190 +
1.191 +TBool CWsSpriteBase::UpdateMemberL(CWsSpriteMember *aMember, const TCmdSpriteMember &aCmdSpriteMember)
1.192 + {
1.193 + CFbsBitmap *old=aMember->iBitmap;
1.194 + CFbsBitmap *oldMask=aMember->iMaskBitmap;
1.195 + aMember->iBitmap=NULL;
1.196 + aMember->iMaskBitmap=NULL;
1.197 + TBool ret=EFalse;
1.198 + TRAPD(err,ret=aMember->SetL(aCmdSpriteMember));
1.199 + if (err!=KErrNone)
1.200 + {
1.201 +um_error:
1.202 + delete aMember->iBitmap;
1.203 + delete aMember->iMaskBitmap;
1.204 + aMember->iBitmap=old;
1.205 + aMember->iMaskBitmap=oldMask;
1.206 + User::Leave(err);
1.207 + }
1.208 + TRAP(err,CheckSizesL());
1.209 + if (err!=KErrNone)
1.210 + goto um_error;
1.211 + SetMember(0);
1.212 + delete old;
1.213 + delete oldMask;
1.214 + return(ret);
1.215 + }
1.216 +
1.217 +void CWsSpriteBase::InitStaticsL()
1.218 + {
1.219 + iDeltaTimer=CWsDeltaTimer::NewL(ESpriteAnimatePriority);
1.220 + }
1.221 +
1.222 +void CWsSpriteBase::DeleteStatics()
1.223 + {
1.224 + delete iDeltaTimer;
1.225 + }
1.226 +
1.227 +CWsSpriteBase::CWsSpriteBase(CWsClient *owner, WH_HANDLES aType) : CWsScreenObject(owner,aType,owner->Screen())
1.228 + {
1.229 + }
1.230 +
1.231 +CWsSpriteBase::~CWsSpriteBase()
1.232 + {
1.233 + Deactivate();
1.234 + if(IsFloating() && iGroupWin)
1.235 + {
1.236 + iGroupWin->RemoveSprite(this);
1.237 + }
1.238 +
1.239 + //iDeltaTimer->Remove(iDeltaTimerEntry);
1.240 + if (iMembers)
1.241 + {
1.242 + iMembers->ResetAndDestroy();
1.243 + delete iMembers;
1.244 + }
1.245 + }
1.246 +
1.247 +void CWsSpriteBase::ForceRedraw()
1.248 + {
1.249 + TRegionFix<1> region;
1.250 + region.AddRect(Rect());
1.251 + Screen()->AddRedrawRegion(region);
1.252 + }
1.253 +
1.254 +void CWsSpriteBase::Deactivate()
1.255 + {
1.256 + //Disconnect from the sprite list and hide the sprite
1.257 + if (iFlags & ESpriteActive)
1.258 + {
1.259 + iFlags&=~ESpriteActive;
1.260 +
1.261 + MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
1.262 + if (windowTreeObserver)
1.263 + {
1.264 + windowTreeObserver->NodeReleased(*this);
1.265 + }
1.266 +
1.267 + if (iMembers && iMembers->Count()>1)
1.268 + {
1.269 + iDeltaTimer->Remove(iDeltaTimerEntry);
1.270 + }
1.271 +
1.272 + if (IsFloating())
1.273 + {
1.274 + Screen()->SpriteManager()->RemoveFloatingSprite(this);
1.275 + if (!Screen()->ChangeTracking())
1.276 + {
1.277 + ForceRedraw();
1.278 + }
1.279 + }
1.280 +
1.281 + // Note: This could be a floating sprite attached to the root window (PDEF138379)
1.282 + if(iWin)
1.283 + {
1.284 + iWin->RemoveSprite(this);
1.285 + }
1.286 + }
1.287 + }
1.288 +
1.289 +/** Called from groupwin destructor only */
1.290 +void CWsSpriteBase::DisconnectGroupWin()
1.291 + {
1.292 + WS_ASSERT_DEBUG(IsFloating(),EWsPanicFloatingSprite);
1.293 + iGroupWin=NULL;
1.294 + }
1.295 +
1.296 +void CWsSpriteBase::CheckSizesL()
1.297 + {
1.298 + iMaxSize.SetSize(0,0);
1.299 + for(TInt index=0;index<iMembers->Count();index++)
1.300 + {
1.301 + CWsSpriteMember *wsm=(*iMembers)[index];
1.302 + if (wsm->iBitmap)
1.303 + {
1.304 + TSize size=wsm->iBitmap->SizeInPixels();
1.305 + if (wsm->iMaskBitmap)
1.306 + {
1.307 + TSize maskSize=wsm->iMaskBitmap->SizeInPixels();
1.308 + if (maskSize.iWidth<size.iWidth || maskSize.iHeight<size.iHeight)
1.309 + OwnerPanic(EWservPanicMaskSize);
1.310 + }
1.311 + if (size.iWidth>iMaxSize.iWidth)
1.312 + iMaxSize.iWidth=size.iWidth;
1.313 + if (size.iHeight>iMaxSize.iHeight)
1.314 + iMaxSize.iHeight=size.iHeight;
1.315 + }
1.316 + }
1.317 + }
1.318 +
1.319 +void CWsSpriteBase::ConstructL(TUint aFlags, CWsWindow *aWindow)
1.320 + {
1.321 + // Common part of construct for both sprites and pointer cursors
1.322 + iFlags=aFlags;
1.323 +/*
1.324 + if (iFlags&ESpriteNoChildClip)
1.325 + iLink.iPriority+=ENoChildPriorityBoost;
1.326 + if (iFlags&ESpritePointer)
1.327 + iLink.iPriority+=EPointerPriorityBoost;
1.328 +*/
1.329 + iWin=aWindow;
1.330 + TCallBack callback(TimerCallBack,this);
1.331 + iDeltaTimerEntry.Set(callback);
1.332 + iMembers=new(ELeave) CArrayPtrFlat<CWsSpriteMember>(10);
1.333 + }
1.334 +
1.335 +void CWsSpriteBase::AppendMemberL(const TCmdSpriteMember &aCmdSpriteMember)
1.336 + {
1.337 + CWsSpriteMember *&pwsm=iMembers->ExtendL();
1.338 + pwsm=NULL; // In case new leaves
1.339 + pwsm=new(ELeave) CWsSpriteMember();
1.340 + // coverity[leave_without_push]
1.341 + // pwsm is not owned by the stack and will be deleted by RWsSpriteBase::Close()
1.342 + if (pwsm->SetL(aCmdSpriteMember))
1.343 + OwnerPanic(EWservPanicBitmap);
1.344 + }
1.345 +
1.346 +void CWsSpriteBase::CompleteL()
1.347 + {
1.348 + if (iMembers->Count() <= 0)
1.349 + {
1.350 + if(iWin)
1.351 + iWin->OwnerPanic(EWservPanicNoSpriteMember);
1.352 + //Not sure if this is a neccessary fall back if iWin is NULL.
1.353 + else if(iWin==NULL && iGroupWin)
1.354 + iGroupWin->OwnerPanic(EWservPanicNoSpriteMember);
1.355 + }
1.356 + else
1.357 + {
1.358 + iMembers->Compress();
1.359 + CheckSizesL();
1.360 + SetMember(0);
1.361 + }
1.362 + }
1.363 +
1.364 +void CWsSpriteBase::Activate()
1.365 + {
1.366 + if (iFlags&ESpriteDisabled)
1.367 + {
1.368 + iFlags&=~ESpriteDisabled;
1.369 + }
1.370 + if (iMembers->Count()>1)
1.371 + {
1.372 + QueueDeltaTimer();
1.373 + iDeltaTimer->Activate();
1.374 + }
1.375 + iFlags|=ESpriteActive;
1.376 + if(iWin)
1.377 + iWin->AddSprite(this);
1.378 + SetDirty(ETrue);
1.379 + Screen()->SpriteManager()->Schedule(this);
1.380 + if(IsFloating())
1.381 + {
1.382 + Screen()->SpriteManager()->AddFloatingSprite(this);
1.383 + if (!Screen()->ChangeTracking())
1.384 + ForceRedraw();
1.385 + }
1.386 +
1.387 +
1.388 + // As custom text cursors are sprites (CWsCustomTextCursor) and can be activated/deactivated
1.389 + // on various different windows during their lifetime, when activating
1.390 + // a text cursor, we pretend it's just been created to give us the option
1.391 + // of specifying a new parent window. Normal sprites (CWsSprite) are
1.392 + // treated the same way just for consistency as it does no harm.
1.393 + MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
1.394 + if (windowTreeObserver)
1.395 + {
1.396 + windowTreeObserver->NodeCreated(*this, ParentNode());
1.397 + windowTreeObserver->NodeExtentChanged(*this, Rect());
1.398 + windowTreeObserver->NodeActivated(*this);
1.399 + }
1.400 + }
1.401 +
1.402 +void CWsSpriteBase::SetMember(TInt aIndex)
1.403 + {
1.404 + const TSize oldSize = iSize;
1.405 + const TPoint oldPos = iPos;
1.406 + if(IsFloating())
1.407 + {
1.408 + TRect rect(iPos,iSize);
1.409 + }
1.410 + iCurIndex=aIndex;
1.411 + iPos=iBasePos+(*iMembers)[iCurIndex]->iOffset;
1.412 + if ((*iMembers)[iCurIndex]->iBitmap)
1.413 + iSize=(*iMembers)[iCurIndex]->iBitmap->SizeInPixels();
1.414 + else
1.415 + iSize.SetSize(0,0);
1.416 +
1.417 + if (iSize.iWidth > iMaxSize.iWidth || iSize.iHeight > iMaxSize.iHeight)
1.418 + {
1.419 + WS_ASSERT_DEBUG(EFalse, EWsPanicSpriteBitmapSizeChange);
1.420 + CheckSizesL();
1.421 + }
1.422 +
1.423 + if(oldSize!=iSize || oldPos!=iPos)
1.424 + NotifyExtentChanged();
1.425 +
1.426 + }
1.427 +
1.428 +void CWsSpriteBase::SetPos(const TPoint &aPos)
1.429 + {
1.430 + //Non-floating anim whose window is destroyed
1.431 + if (!IsFloating() && iWin==NULL)
1.432 + {
1.433 + OwnerPanic(EWservPanicWindowDestroyed);
1.434 + }
1.435 +
1.436 + //Floating anim whose group window is destroyed
1.437 + if (IsFloating() && iGroupWin==NULL && iWin->WinType() != EWinTypeRoot)
1.438 + {
1.439 + OwnerPanic(EWservPanicWindowDestroyed);
1.440 + }
1.441 +
1.442 + if(IsFloating())
1.443 + {
1.444 + TRect rect(iPos,iSize);
1.445 + }
1.446 +
1.447 + iBasePos=aPos;
1.448 + TPoint newPos(iBasePos+(*iMembers)[iCurIndex]->iOffset);
1.449 + if (iPos!=newPos)
1.450 + {
1.451 + if (!Screen()->ChangeTracking())
1.452 + //Ensure the region covered by the sprite before as well as after the move gets scheduled for redraw
1.453 + Screen()->SpriteManager()->Schedule(this);
1.454 + iPos=newPos;
1.455 + if (!Screen()->ChangeTracking())
1.456 + Screen()->SpriteManager()->Schedule(this);
1.457 + NotifyExtentChanged();
1.458 + }
1.459 + }
1.460 +
1.461 +void CWsSpriteBase::QueueDeltaTimer()
1.462 + {
1.463 + iDeltaTimer->Queue((*iMembers)[iCurIndex]->iInterval,iDeltaTimerEntry);
1.464 + }
1.465 +
1.466 +void CWsSpriteBase::TimerExpired()
1.467 + {
1.468 + if (!Screen()->ChangeTracking())
1.469 + Screen()->SpriteManager()->Schedule(this);
1.470 + SetMember((iCurIndex+1)==iMembers->Count() ? 0 : iCurIndex+1);
1.471 + SetDirty(ETrue);
1.472 + Screen()->SpriteManager()->Schedule(this);
1.473 + QueueDeltaTimer();
1.474 + }
1.475 +
1.476 +TPoint CWsSpriteBase::Pos() const
1.477 + {
1.478 + if (iGroupWin || iWin==NULL || iWin->WinType() == EWinTypeRoot )
1.479 + {
1.480 + return(iPos);
1.481 + }
1.482 + return(iPos+iWin->Origin());
1.483 + }
1.484 +
1.485 +TRect CWsSpriteBase::Rect() const
1.486 + {
1.487 + TRect rect;
1.488 + rect.iTl=Pos();
1.489 + rect.iBr=rect.iTl+iSize;
1.490 + return(rect);
1.491 + }
1.492 +
1.493 +MWsSprite::TSpriteType CWsSpriteBase::SpriteType() const
1.494 + {
1.495 + if (IsFloating())
1.496 + return EFloatingSprite;
1.497 +
1.498 + TSpriteType spriteType = EWindowSprite;
1.499 + switch(Type())
1.500 + {
1.501 + case WS_HANDLE_SPRITE:
1.502 + spriteType = EWindowSprite;
1.503 + break;
1.504 + case WS_HANDLE_POINTER_CURSOR:
1.505 + spriteType = EPointerCursorSprite;
1.506 + break;
1.507 + case WS_HANDLE_TEXT_CURSOR:
1.508 + spriteType = ECustomTextCursorSprite;
1.509 + break;
1.510 + default:
1.511 + ASSERT(0);
1.512 + }
1.513 + return spriteType;
1.514 + }
1.515 +
1.516 +void CWsSpriteBase::CommandL(TInt aOpcode, const TAny *aCmdData)
1.517 + {
1.518 + TWsSpriteCmdUnion pData;
1.519 +
1.520 + pData.any=aCmdData;
1.521 + switch(aOpcode)
1.522 + {
1.523 + case EWsSpriteOpAppendMember:
1.524 + AppendMemberL(*pData.SpriteMember);
1.525 + break;
1.526 + case EWsSpriteOpActivate:
1.527 + if(!(iFlags&ESpriteActive))
1.528 + CompleteL();
1.529 + break;
1.530 + case EWsSpriteOpUpdateMember:
1.531 + if (pData.UpdateMember->index==iCurIndex)
1.532 + {
1.533 + SetDirty(ETrue);
1.534 + TRect rect(Pos(), iMaxSize);
1.535 + Screen()->SpriteManager()->Schedule(this,&rect);
1.536 + }
1.537 + break;
1.538 + case EWsSpriteOpUpdateMember2:
1.539 + {
1.540 + SetDirty(ETrue);
1.541 + Screen()->SpriteManager()->Schedule(this);
1.542 + if (pData.UpdateMember->index<0 || pData.UpdateMember->index>=iMembers->Count())
1.543 + User::Leave(KErrArgument);
1.544 + CWsSpriteMember *member=(*iMembers)[pData.UpdateMember->index];
1.545 + TBool ret=EFalse;
1.546 + TRAPD(err,ret=UpdateMemberL(member,pData.UpdateMember->data));
1.547 + if (err==KErrNone)
1.548 + {
1.549 + TRAP(err,CheckSizesL());
1.550 + SetMember(0);
1.551 + }
1.552 + Screen()->SpriteManager()->Schedule(this);
1.553 + User::LeaveIfError(err);
1.554 + if (ret)
1.555 + OwnerPanic(EWservPanicBitmap);
1.556 + }
1.557 + break;
1.558 + default:
1.559 + OwnerPanic(EWservPanicOpcode);
1.560 + break;
1.561 + }
1.562 + }
1.563 +
1.564 +TBool CWsSpriteBase::CanBeSeen() const
1.565 + {
1.566 + if(iWin)
1.567 + return (!(iFlags&ESpriteDisabled)) && (!iWin->VisibleRegion().IsEmpty());
1.568 + else
1.569 + return (!(iFlags&ESpriteDisabled));
1.570 + }
1.571 +
1.572 +void CWsSpriteBase::CalcRedrawRegion(const TRegion& aSourceRegion, TRegion& aTarget) const
1.573 + {
1.574 + aTarget.Copy(aSourceRegion);
1.575 + if (ClipSprite())
1.576 + {
1.577 + TPoint origin(0,0);
1.578 + if(iWin)
1.579 + origin = iWin->Origin();
1.580 + TRect rect(iBasePos + origin + iClipOffset, iClipSize);
1.581 + aTarget.ClipRect(rect);
1.582 + }
1.583 + aTarget.ClipRect(RootWindow()->Abs());
1.584 +
1.585 + // Only need to draw if the region being redrawn overlaps the sprite
1.586 + const TRect spriteRect(Pos(), iSize);
1.587 + STACK_REGION spriteRegion;
1.588 + spriteRegion.AddRect(spriteRect);
1.589 + aTarget.Intersect(spriteRegion);
1.590 + spriteRegion.Close();
1.591 + }
1.592 +
1.593 +/**
1.594 + @pre CWsSpriteBase::CalcRedrawRegion(TRegion&,TRegion&) should have been called.
1.595 + @param aRegion Is the region that will definitely be redrawn if dirty. This param should
1.596 + be calculated by calling CalcRedrawRegion(TRegion&,TRegion&)
1.597 + */
1.598 +void CWsSpriteBase::Redraw(MWsGraphicsContext * aGc, const TRegion& aRegion)
1.599 + {
1.600 + TFlashState currentState=EFlashOn;
1.601 + if(IsFlashingEnabled())
1.602 + {
1.603 + currentState=Screen()->SpriteManager()->CurrentSpriteFlashState(this);
1.604 + AnnotateSpriteFlash(*this, currentState==EFlashOn);
1.605 + }
1.606 +
1.607 + if(currentState==EFlashOn && (IsDirty() || HasAnimation()) )
1.608 + {
1.609 + const TRegion * pr = &aRegion;
1.610 +
1.611 + if (pr->CheckError())
1.612 + {
1.613 + if(iWin)
1.614 + {
1.615 + if (Screen()->ChangeTracking())
1.616 + pr = &iWin->WindowArea();
1.617 + else
1.618 + pr = &iWin->VisibleRegion();
1.619 + }
1.620 + else
1.621 + pr = &RootWindow()->WindowArea();
1.622 + }
1.623 +
1.624 + if (!pr->IsEmpty())
1.625 + {
1.626 + CWsSpriteMember *member=(*iMembers)[iCurIndex];
1.627 + if (member->iBitmap)
1.628 + {
1.629 + aGc->SetClippingRegion(*pr);
1.630 +
1.631 + // Calculate which piece (rect) of the bitmap needs to be drawn
1.632 + const TRect redrawRect = pr->BoundingRect();
1.633 + TRect bitmapRect(Pos(), iSize); // sprite rect relative to screen
1.634 + bitmapRect.Intersection(redrawRect);
1.635 + bitmapRect.Move(-Pos()); // adjust relative to bitmap origin
1.636 +
1.637 + if (member->iMaskBitmap)
1.638 + aGc->BitBltMasked(Pos() + bitmapRect.iTl, *member->iBitmap, bitmapRect, *member->iMaskBitmap, member->iInvertMask);
1.639 + else
1.640 + {
1.641 + aGc->SetDrawMode(BitGdiToMWsGraphicsContextMappings::LossyConvert(member->iDrawMode));
1.642 + aGc->BitBlt(Pos() + bitmapRect.iTl, *member->iBitmap, bitmapRect);
1.643 + aGc->SetDrawMode(MWsGraphicsContext::EDrawModePEN);
1.644 + }
1.645 + aGc->ResetClippingRegion();
1.646 + }
1.647 + }
1.648 + if (Screen()->ChangeTracking())
1.649 + SetDirty(EFalse);
1.650 +
1.651 + }
1.652 +
1.653 + //Flashing sprites need to reschedule themselves after drawing (unless they have
1.654 + //an animation, because for animating sprites the rescheduling is done in CWsAnim).
1.655 + if(IsFlashingEnabled() && !HasAnimation())
1.656 + Screen()->SpriteManager()->Schedule(this);
1.657 + }
1.658 +
1.659 +TBool CWsSpriteBase::IsActivated() const
1.660 + {
1.661 + return(iFlags&ESpriteActive);
1.662 + }
1.663 +
1.664 +void CWsSpriteBase::SendState(MWsWindowTreeObserver& aWindowTreeObserver) const
1.665 + {
1.666 + if(iNext)
1.667 + iNext->SendState(aWindowTreeObserver);
1.668 +
1.669 + if(IsActivated())
1.670 + {
1.671 + //Sprite NodeCreated must only be sent if activated
1.672 + aWindowTreeObserver.NodeCreated(*this, ParentNode());
1.673 + aWindowTreeObserver.NodeExtentChanged(*this, Rect());
1.674 + aWindowTreeObserver.NodeActivated(*this);
1.675 + }
1.676 + }
1.677 +
1.678 +/** @see MWsWindowTreeNode */
1.679 +MWsWindowTreeNode::TType CWsSpriteBase::NodeType() const
1.680 + {
1.681 + return MWsWindowTreeNode::EWinTreeNodeSprite;
1.682 + }
1.683 +
1.684 +/** @see MWsWindowTreeNode */
1.685 +const MWsWindow* CWsSpriteBase::Window() const
1.686 + {
1.687 + return NULL;
1.688 + }
1.689 +
1.690 +/** @see MWsWindowTreeNode */
1.691 +const MWsSprite* CWsSpriteBase::Sprite() const
1.692 + {
1.693 + return this;
1.694 + }
1.695 +
1.696 +/** @see MWsWindowTreeNode */
1.697 +const MWsStandardTextCursor* CWsSpriteBase::StandardTextCursor() const
1.698 + {
1.699 + return NULL;
1.700 + }
1.701 +
1.702 +/** @see MWsWindowTreeNode */
1.703 +const MWsWindowGroup* CWsSpriteBase::WindowGroup() const
1.704 + {
1.705 + if(iWin)
1.706 + return iWin->WindowGroup();
1.707 + else if (iGroupWin)
1.708 + return static_cast<MWsWindowGroup*>(iGroupWin); //floating Sprite
1.709 +
1.710 + WS_ASSERT_DEBUG(EFalse,EWsPanicInvalidOperation);
1.711 + return NULL;
1.712 + }
1.713 +
1.714 +/** @see MWsWindowTreeNode */
1.715 +const MWsWindowTreeNode* CWsSpriteBase::ParentNode() const
1.716 + {
1.717 + if (iWin)
1.718 + return iWin;
1.719 + else if (iGroupWin)
1.720 + return iGroupWin->BaseParent(); //floating Sprite, will return the rootwin
1.721 +
1.722 + WS_ASSERT_DEBUG(EFalse,EWsPanicInvalidOperation);
1.723 + return NULL;
1.724 + }
1.725 +
1.726 +void CWsSpriteBase::NotifyExtentChanged() const
1.727 + {
1.728 + if (Screen())
1.729 + {
1.730 + MWsWindowTreeObserver* windowTreeObserver = Screen()->WindowTreeObserver();
1.731 + if (windowTreeObserver && iFlags&ESpriteActive)
1.732 + {
1.733 + windowTreeObserver->NodeExtentChanged(*this, Rect());
1.734 + }
1.735 + }
1.736 + }
1.737 +
1.738 +//
1.739 +// CWsSprite
1.740 +//
1.741 +
1.742 +CWsSprite::CWsSprite(CWsClient *owner) : CWsSpriteBase(owner,WS_HANDLE_SPRITE)
1.743 + {
1.744 + }
1.745 +
1.746 +CWsSprite::~CWsSprite()
1.747 + {
1.748 + if (!IsFloating() && IsActivated() && iWin && iWin->IsVisible() && !Screen()->ChangeTracking())
1.749 + ForceRedraw();
1.750 + if (iAnim)
1.751 + CWsAnim::CloseAnim(iAnim);
1.752 + }
1.753 +
1.754 +void CWsSprite::ConstructL(const TWsClCmdCreateSprite &aParams)
1.755 + {
1.756 + NewObjL();
1.757 + CWsWindowBase *win;
1.758 + WsOwner()->HandleToWindow(aParams.window,&win);
1.759 + WS_ASSERT_DEBUG(win,EWsPanicWindowNull);
1.760 + if (win->WinType()==EWinTypeGroup)
1.761 + {
1.762 + //If a sprite is attached to a group window it is floating.
1.763 + //Floating sprite drawing is performed by the sprite manager.
1.764 + iGroupWin=(CWsWindowGroup *)win;
1.765 + win=NULL; //Floating sprites aren't associated with any particular window.
1.766 + SetIsFloating(ETrue);
1.767 + //In case the GroupWin is destroyed before the sprite it needs to call DisconnectGroupWin
1.768 + iGroupWin->AddSprite(this);
1.769 + }
1.770 + CWsSpriteBase::ConstructL(aParams.flags&ESpriteNonSystemFlags,(CWsWindow *)win);
1.771 + iBasePos=aParams.pos;
1.772 + }
1.773 +
1.774 +void CWsSprite::CompleteL()
1.775 + {
1.776 + CWsSpriteBase::CompleteL();
1.777 + Activate();
1.778 + }
1.779 +
1.780 +void CWsSprite::CommandL(TInt aOpcode, const TAny *aCmdData)
1.781 + {
1.782 + TWsSpriteCmdUnion pData;
1.783 + pData.any=aCmdData;
1.784 + switch(aOpcode)
1.785 + {
1.786 + case EWsSpriteOpSetPosition:
1.787 + SetPos(*pData.Point);
1.788 + break;
1.789 + case EWsSpriteOpFree:
1.790 + {
1.791 + delete this;
1.792 + break;
1.793 + }
1.794 + default:
1.795 + CWsSpriteBase::CommandL(aOpcode, aCmdData);
1.796 + break;
1.797 + }
1.798 + }
1.799 +
1.800 +/**
1.801 +@see MAnimSpriteFunctions::UpdateMember
1.802 +@param aFullUpdate Not used. Wserv2 always do full back to front rendering, so there is no distinction between changes needing aFullUpdate or not
1.803 + */
1.804 +void CWsSprite::Update(TInt aMember,TRect aRect,TBool /*aFullUpdate*/)
1.805 + {
1.806 + if (iCurIndex!=aMember)
1.807 + return;
1.808 + aRect.Move(Pos());
1.809 + aRect.Intersection(iScreen->CurrentScreenSize());
1.810 + SetDirty(ETrue);
1.811 + Screen()->SpriteManager()->Schedule(this, &aRect);
1.812 + }
1.813 +
1.814 +//
1.815 +// CWsPointerCursor
1.816 +//
1.817 +
1.818 +CWsPointerCursor::CWsPointerCursor(CWsClient *owner) : CWsSpriteBase(owner,WS_HANDLE_POINTER_CURSOR)
1.819 + {
1.820 + }
1.821 +
1.822 +void CWsPointerCursor::CloseObject()
1.823 + {
1.824 + RemoveFromIndex();
1.825 + Close();
1.826 + }
1.827 +
1.828 +void CWsPointerCursor::Close()
1.829 + {
1.830 + WS_ASSERT_DEBUG(iAccessCount>0, EWsPanicPointerCursorAccessCount);
1.831 + if (--iAccessCount==0)
1.832 + delete this;
1.833 + }
1.834 +
1.835 +void CWsPointerCursor::Open()
1.836 + {
1.837 + iAccessCount++;
1.838 + }
1.839 +
1.840 +CWsPointerCursor::~CWsPointerCursor()
1.841 + {
1.842 + WS_ASSERT_DEBUG(iAccessCount==0, EWsPanicPointerCursorAccessCount);
1.843 + }
1.844 +
1.845 +void CWsPointerCursor::ConstructL(const TWsClCmdCreatePointerCursor &aParams)
1.846 + {
1.847 + NewObjL();
1.848 + SetIsFloating(ETrue);
1.849 + CWsSpriteBase::ConstructL(ESpriteNoShadows|ESpriteNoChildClip|ESpritePointer|(aParams.flags&ESpriteNonSystemFlags),RootWindow());
1.850 + Open();
1.851 + }
1.852 +
1.853 +void CWsPointerCursor::CommandL(TInt aOpcode, const TAny *aCmdData)
1.854 + {
1.855 + switch(aOpcode)
1.856 + {
1.857 + case EWsSpriteOpFree:
1.858 + CloseObject();
1.859 + break;
1.860 + default:
1.861 + CWsSpriteBase::CommandL(aOpcode, aCmdData);
1.862 + break;
1.863 + }
1.864 + }
1.865 +
1.866 +//
1.867 +// CWsCustomTextCursor
1.868 +//
1.869 +
1.870 +CWsCustomTextCursor::CWsCustomTextCursor (CWsClient *aOwner, RWsSession::TCustomTextCursorAlignment aAlignment)
1.871 +: CWsSpriteBase(aOwner, WS_HANDLE_TEXT_CURSOR), iAlignment(aAlignment)
1.872 + {
1.873 + }
1.874 +
1.875 +CWsCustomTextCursor::~CWsCustomTextCursor()
1.876 + {
1.877 + }
1.878 +
1.879 +void CWsCustomTextCursor::ConstructL(TInt aFlags)
1.880 + {
1.881 + NewObjL();
1.882 + CWsSpriteBase::ConstructL(ESpriteNoShadows|ESpriteNoChildClip|ESpritePointer|(aFlags&ESpriteNonSystemFlags), NULL);
1.883 + }
1.884 +
1.885 +void CWsCustomTextCursor::CompleteL(CWsWindow *aWin, TBool aFlash, TBool aClipSprite, const TPoint& aClipOffset, const TSize& aClipSize)
1.886 + {
1.887 + iWin = aWin;
1.888 + iFlags = aFlash ? iFlags | ESpriteFlash : iFlags & ~ESpriteFlash;
1.889 + SetClipSprite(aClipSprite);
1.890 + iClipOffset = aClipOffset;
1.891 + iClipSize = aClipSize;
1.892 + CWsSpriteBase::CompleteL();
1.893 + }
1.894 +
1.895 +// Use SetPositionNoRedraw instead of SetPos when you just want to update
1.896 +// the custom text cursor position without redrawing it
1.897 +void CWsCustomTextCursor::SetPositionNoRedraw(const TPoint& aPos)
1.898 + {
1.899 + iBasePos = aPos;
1.900 + TPoint newPos(iBasePos+(*iMembers)[iCurIndex]->iOffset);
1.901 + iPos=newPos;
1.902 + }
1.903 +
1.904 +void CWsCustomTextCursor::CommandL(TInt aOpcode, const TAny *aCmdData)
1.905 + {
1.906 + switch(aOpcode)
1.907 + {
1.908 + case EWsSpriteOpFree:
1.909 + // CWsCustomTextCursor objects are owned by the text cursor list.
1.910 + // They are not deleted when the client closes it's R class.
1.911 + RemoveFromIndex();
1.912 + break;
1.913 + default:
1.914 + CWsSpriteBase::CommandL(aOpcode, aCmdData);
1.915 + break;
1.916 + }
1.917 + }
1.918 +
1.919 +//
1.920 +// CWsDeltaTimer, nicked from CDeltaTimer and tweaked so it doesn't re-activate //
1.921 +// the timers until RunL has finished running all ready timers. //
1.922 +// //
1.923 +// This is to stop a problem in Wserv where sprites could hog 100% CPU if the time //
1.924 +// it took to process them was longer than the time of the timer queued when the first //
1.925 +// sprite was updated //
1.926 +//
1.927 +
1.928 +CWsDeltaTimer* CWsDeltaTimer::NewL(TInt aPriority)
1.929 + {
1.930 + CWsDeltaTimer* wsdt=new(ELeave) CWsDeltaTimer(aPriority);
1.931 + CleanupStack::PushL(wsdt);
1.932 + User::LeaveIfError(wsdt->iTimer.CreateLocal());
1.933 + CActiveScheduler::Add(wsdt);
1.934 + CleanupStack::Pop(wsdt);
1.935 + return(wsdt);
1.936 + }
1.937 +
1.938 +CWsDeltaTimer::CWsDeltaTimer(TInt aPriority) : CActive(aPriority),iQueue(_FOFF(TWsDeltaTimerEntry,iLink))
1.939 + {
1.940 + }
1.941 +
1.942 +CWsDeltaTimer::~CWsDeltaTimer()
1.943 + {
1.944 + Cancel();
1.945 + while(iQueue.RemoveFirst()!=NULL)
1.946 + {}
1.947 + }
1.948 +
1.949 +void CWsDeltaTimer::Queue(TTimeIntervalMicroSeconds32 aTimeInMicroSeconds,TWsDeltaTimerEntry& anEntry)
1.950 + {
1.951 + TInt intervals=aTimeInMicroSeconds.Int()/CWsDeltaTimerGranularity;
1.952 + if (intervals==0)
1.953 + intervals=1;
1.954 + iQueue.Add(anEntry,intervals);
1.955 + }
1.956 +
1.957 +void CWsDeltaTimer::Activate()
1.958 + {
1.959 + // Queue a request on the timer.
1.960 + // The timer runs every tenth of a second and decremented the delta of the head of the queue.
1.961 + if (IsActive())
1.962 + return;
1.963 + if (!iQueue.IsEmpty())
1.964 + {
1.965 + SetActive();
1.966 + iTimer.After(iStatus,CWsDeltaTimerGranularity-1); // -1 to compensate for +1 in kernel!
1.967 + }
1.968 + }
1.969 +
1.970 +void CWsDeltaTimer::RunL()
1.971 + {
1.972 + // Call all zero delta callbacks
1.973 + iQueue.CountDown();
1.974 + TWsDeltaTimerEntry* ent=iQueue.RemoveFirst();
1.975 + while (ent)
1.976 + {
1.977 + ent->iCallBack.CallBack();
1.978 + ent=iQueue.RemoveFirst();
1.979 + }
1.980 + Activate();
1.981 + }
1.982 +
1.983 +void CWsDeltaTimer::DoCancel()
1.984 + {
1.985 + iTimer.Cancel();
1.986 + }
1.987 +
1.988 +void CWsDeltaTimer::Remove(TWsDeltaTimerEntry& anEntry)
1.989 + {
1.990 + if (anEntry.IsPending())
1.991 + {
1.992 + iQueue.Remove(anEntry);
1.993 + Activate();
1.994 + }
1.995 + }
1.996 +
1.997 +
1.998 +//
1.999 +// CWsSpriteManager -handles floating and flashing sprites including flashing custom text cursors. i.e. cursors
1.1000 +// that have an associated sprite.
1.1001 +
1.1002 +
1.1003 +CWsSpriteManager::CWsSpriteManager()
1.1004 + {
1.1005 + }
1.1006 +
1.1007 +CWsSpriteManager* CWsSpriteManager::NewL()
1.1008 + {
1.1009 + CWsSpriteManager* self = new (ELeave) CWsSpriteManager();
1.1010 + CleanupStack::PushL(self);
1.1011 + self->ConstructL();
1.1012 + CleanupStack::Pop(self);
1.1013 + return self;
1.1014 + }
1.1015 +
1.1016 +void CWsSpriteManager::ConstructL()
1.1017 + {
1.1018 + }
1.1019 +
1.1020 +CWsSpriteManager::~CWsSpriteManager()
1.1021 + {
1.1022 + iFloatingSprites.ResetAndDestroy();
1.1023 + }
1.1024 +void CWsSpriteManager::AddFloatingSprite(CWsSpriteBase* aSprite)
1.1025 + {
1.1026 + iFloatingSprites.Append(aSprite);
1.1027 + }
1.1028 +
1.1029 +void CWsSpriteManager::RemoveFloatingSprite(CWsSpriteBase* aSprite)
1.1030 + {
1.1031 +#ifdef _DEBUG
1.1032 + TInt removed = 0;
1.1033 +#endif
1.1034 + for (TInt i=0 ; i<iFloatingSprites.Count() ; i++)
1.1035 + {
1.1036 + if(iFloatingSprites[i]==aSprite)
1.1037 + {
1.1038 + //Just remove the sprite don't delete it. the manager doesn't have ownership
1.1039 + iFloatingSprites.Remove(i);
1.1040 +#ifdef _DEBUG
1.1041 + removed++;
1.1042 +#else
1.1043 + break;
1.1044 +#endif
1.1045 + }
1.1046 + }
1.1047 + WS_ASSERT_DEBUG(removed==1,EWsPanicFloatingSprite);
1.1048 + }
1.1049 +
1.1050 +void CWsSpriteManager::DrawFloatingSprites(MWsGraphicsContext* aGc,const TRegion& aRegion)
1.1051 + {
1.1052 + if (iFloatingSprites.Count() == 0)
1.1053 + return; //avoid sending events unless necessary
1.1054 +
1.1055 + for (TInt i = iFloatingSprites.Count() - 1; i >= 0 ; i--)
1.1056 + {
1.1057 + STACK_REGION redrawRegion;
1.1058 + CWsSpriteBase* sprite = iFloatingSprites[i];
1.1059 + sprite->CalcRedrawRegion(aRegion, redrawRegion);
1.1060 + if(redrawRegion.CheckError() || !redrawRegion.IsEmpty())
1.1061 + {
1.1062 + if (sprite->IsFlashingEnabled() || sprite->IsDirty() || sprite->HasAnimation())
1.1063 + {
1.1064 + AnnotateSpriteRedrawStart(*sprite, redrawRegion);
1.1065 +
1.1066 + if(sprite->HasAnimation())
1.1067 + {
1.1068 + CWsAnim* anim = static_cast<CWsSprite*>(sprite)->iAnim;
1.1069 + ASSERT(anim);
1.1070 +
1.1071 + TRAPD(err, anim->AnimateSpriteAnimL(sprite->Screen()->Now()));
1.1072 + if(err!=KErrNone)
1.1073 + {
1.1074 + AnnotateSpriteRedrawEnd(*sprite);
1.1075 + anim->Panic(EWservPanicAnimLeave);
1.1076 + continue;
1.1077 + }
1.1078 + }
1.1079 +
1.1080 + aGc->Reset();
1.1081 + sprite->Redraw(aGc, redrawRegion);
1.1082 +
1.1083 + AnnotateSpriteRedrawEnd(*sprite);
1.1084 + }
1.1085 + }
1.1086 + redrawRegion.Close();
1.1087 + }
1.1088 + }
1.1089 +
1.1090 +void CWsSpriteManager::Schedule(CWsSpriteBase* aSprite, TRect* aRect)
1.1091 + {
1.1092 + if (aRect != NULL && aRect->IsEmpty())
1.1093 + return;
1.1094 +
1.1095 + TRect rect = aSprite->Rect();
1.1096 + if (aRect)
1.1097 + rect.Intersection(*aRect);
1.1098 +
1.1099 + const TAnimType type = aSprite->Win() ? EWindowSprite : EFloatingSprite;
1.1100 +
1.1101 + if(aSprite->IsFlashingEnabled())
1.1102 + {
1.1103 + aSprite->Screen()->ScheduleAnimation(type, rect,NextSpriteFlashStateChange(aSprite),0,0, aSprite->Win());
1.1104 + }
1.1105 + else
1.1106 + {
1.1107 + //Scheduling an animation "now" means it will take place at next animation which might
1.1108 + //be the full animation grace period into the future (see KAnimationGrace in server.cpp)
1.1109 + aSprite->Screen()->ScheduleAnimation(type, rect,0,0,0, aSprite->Win());
1.1110 + }
1.1111 + }
1.1112 +
1.1113 +// Sprite flashing is clamped to half second intervals in relation to the global time.
1.1114 +// For the first half of each second all sprites have the EFlashOn state (visible)
1.1115 +// For the second half of each second all sprites have the EFlashOff state (not visible)
1.1116 +TTimeIntervalMicroSeconds CWsSpriteManager::NextSpriteFlashStateChange(const CWsSpriteBase* aSprite) const
1.1117 + {
1.1118 + const TTimeIntervalMicroSeconds remainder = aSprite->Screen()->Now().DateTime().MicroSecond();
1.1119 + return CalculateTimeToNextFlash(remainder);
1.1120 + }
1.1121 +
1.1122 +TTimeIntervalMicroSeconds CWsSpriteManager::NextCursorFlashStateChange() const
1.1123 + {
1.1124 + const TTimeIntervalMicroSeconds remainder = CWsTop::CurrentFocusScreen()->Now().DateTime().MicroSecond();
1.1125 + return CalculateTimeToNextFlash(remainder);
1.1126 + }
1.1127 +
1.1128 +TTimeIntervalMicroSeconds CWsSpriteManager::CalculateTimeToNextFlash(TTimeIntervalMicroSeconds aTime) const
1.1129 + {
1.1130 + TInt64 nextStateChange;
1.1131 + if(aTime<KFlashHalfSecond)
1.1132 + nextStateChange=KFlashHalfSecond-aTime.Int64();
1.1133 + else
1.1134 + nextStateChange=KFlashHalfSecond - (aTime.Int64() - KFlashHalfSecond);
1.1135 + ASSERT(nextStateChange > 0);
1.1136 + return TTimeIntervalMicroSeconds(nextStateChange);
1.1137 + }
1.1138 +
1.1139 +TFlashState CWsSpriteManager::CurrentSpriteFlashState(const CWsSpriteBase* aSprite) const
1.1140 + {
1.1141 + return (aSprite->Screen()->Now().DateTime().MicroSecond()<KFlashHalfSecond)?EFlashOn:EFlashOff;
1.1142 + }
1.1143 +
1.1144 +void CWsSpriteManager::CalcFloatingSpriteRgn( TRegion& aResultRgn, const TRect& aDefaultRect )
1.1145 + {
1.1146 + aResultRgn.Clear();
1.1147 + for (TInt i=0 ; i<iFloatingSprites.Count() && !aResultRgn.CheckError(); i++)
1.1148 + {
1.1149 + CWsSpriteBase* sprite = iFloatingSprites[i];
1.1150 + if ( sprite->CanBeSeen() && ( sprite->IsActive() || sprite->IsActivated() ) )
1.1151 + {
1.1152 + aResultRgn.AddRect( sprite->Rect() );
1.1153 + }
1.1154 + }
1.1155 + aResultRgn.Tidy();
1.1156 + if ( aResultRgn.CheckError() && iFloatingSprites.Count() > 0 )
1.1157 + {
1.1158 + aResultRgn.Clear();
1.1159 + aResultRgn.AddRect( aDefaultRect );
1.1160 + }
1.1161 + }
1.1162 +
1.1163 +void CWsSpriteManager::SendState(MWsWindowTreeObserver& aWindowTreeObserver) const
1.1164 + {
1.1165 + for(TInt i=iFloatingSprites.Count()-1; i>=0; i--)
1.1166 + {
1.1167 + CWsSpriteBase* sprite = iFloatingSprites[i];
1.1168 + sprite->SendState(aWindowTreeObserver);
1.1169 + }
1.1170 + }