1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nonnga/SERVER/SPRITE.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,838 @@
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 "offscreenbitmap.h"
1.29 +#include "panics.h"
1.30 +#include "EVENT.H"
1.31 +
1.32 +static const TInt64 KFlashHalfSecond(500000); //interval for flashing sprites
1.33 +
1.34 +GLDEF_D CWsDeltaTimer *CWsSpriteBase::iDeltaTimer=NULL;
1.35 +
1.36 +TInt TimerCallBack(TAny *aPtr)
1.37 + {
1.38 + ((CWsSpriteBase *)aPtr)->TimerExpired();
1.39 + return(KErrNone);
1.40 + }
1.41 +
1.42 +//
1.43 +// CWsSpriteMember
1.44 +//
1.45 +
1.46 +CWsSpriteMember::CWsSpriteMember()
1.47 + {
1.48 + }
1.49 +
1.50 +CWsSpriteMember::~CWsSpriteMember()
1.51 + {
1.52 + delete iBitmap;
1.53 + delete iMaskBitmap;
1.54 + }
1.55 +
1.56 +TBool CWsSpriteMember::SetL(const TCmdSpriteMember &aCmdSpriteMember)
1.57 + {
1.58 + if (aCmdSpriteMember.iBitmap!=0) // Null member of sequence, time is only valid field in member data
1.59 + {
1.60 + iBitmap=new(ELeave) CFbsBitmap();
1.61 + TInt ret = iBitmap->Duplicate(aCmdSpriteMember.iBitmap);
1.62 + if (ret == KErrNoMemory)
1.63 + {
1.64 + User::Leave(ret);
1.65 + }
1.66 + if (ret != KErrNone)
1.67 + return(ETrue);
1.68 +
1.69 + if (aCmdSpriteMember.iMaskBitmap)
1.70 + {
1.71 + iMaskBitmap=new(ELeave) CFbsBitmap();
1.72 + TInt ret = iMaskBitmap->Duplicate(aCmdSpriteMember.iMaskBitmap);
1.73 + if (ret == KErrNoMemory)
1.74 + {
1.75 + User::Leave(ret);
1.76 + }
1.77 + if (ret != KErrNone)
1.78 + return(ETrue);
1.79 + }
1.80 + iInvertMask=aCmdSpriteMember.iInvertMask;
1.81 + iDrawMode=aCmdSpriteMember.iDrawMode;
1.82 + iOffset=aCmdSpriteMember.iOffset;
1.83 + }
1.84 + iInterval=aCmdSpriteMember.iInterval;
1.85 + return(EFalse);
1.86 + }
1.87 +
1.88 +//
1.89 +// CWsSpriteBase
1.90 +//
1.91 +
1.92 +TBool CWsSpriteBase::UpdateMemberL(CWsSpriteMember *aMember, const TCmdSpriteMember &aCmdSpriteMember)
1.93 + {
1.94 + CFbsBitmap *old=aMember->iBitmap;
1.95 + CFbsBitmap *oldMask=aMember->iMaskBitmap;
1.96 + aMember->iBitmap=NULL;
1.97 + aMember->iMaskBitmap=NULL;
1.98 + TBool ret=EFalse;
1.99 + TRAPD(err,ret=aMember->SetL(aCmdSpriteMember));
1.100 + if (err!=KErrNone)
1.101 + {
1.102 +um_error:
1.103 + delete aMember->iBitmap;
1.104 + delete aMember->iMaskBitmap;
1.105 + aMember->iBitmap=old;
1.106 + aMember->iMaskBitmap=oldMask;
1.107 + User::Leave(err);
1.108 + }
1.109 + TRAP(err,CheckSizesL());
1.110 + if (err!=KErrNone)
1.111 + goto um_error;
1.112 + SetMember(0);
1.113 + delete old;
1.114 + delete oldMask;
1.115 + return(ret);
1.116 + }
1.117 +
1.118 +void CWsSpriteBase::InitStaticsL()
1.119 + {
1.120 + iDeltaTimer=CWsDeltaTimer::NewL(ESpriteAnimatePriority);
1.121 + }
1.122 +
1.123 +void CWsSpriteBase::DeleteStatics()
1.124 + {
1.125 + delete iDeltaTimer;
1.126 + }
1.127 +
1.128 +CWsSpriteBase::CWsSpriteBase(CWsClient *owner, WH_HANDLES aType) : CWsScreenObject(owner,aType,owner->Screen()), iClipSprite(EFalse)
1.129 + {
1.130 + }
1.131 +
1.132 +CWsSpriteBase::~CWsSpriteBase()
1.133 + {
1.134 + Deactivate();
1.135 +
1.136 + //iDeltaTimer->Remove(iDeltaTimerEntry);
1.137 + if (iMembers)
1.138 + {
1.139 + iMembers->ResetAndDestroy();
1.140 + delete iMembers;
1.141 + }
1.142 + }
1.143 +
1.144 +void CWsSpriteBase::ForceRedraw()
1.145 + {
1.146 + TRegionFix<1> region;
1.147 + region.AddRect(Rect());
1.148 + Screen()->AddRedrawRegion(region);
1.149 + }
1.150 +
1.151 +void CWsSpriteBase::Deactivate()
1.152 + {
1.153 + //Disconnect from the sprite list and hide the sprite
1.154 + if (iFlags & ESpriteActive)
1.155 + {
1.156 + if(iWin)
1.157 + iWin->RemoveSprite(this);
1.158 + if (iMembers && iMembers->Count()>1)
1.159 + iDeltaTimer->Remove(iDeltaTimerEntry);
1.160 + iFlags&=~ESpriteActive;
1.161 + if (iFloating)
1.162 + {
1.163 + Screen()->SpriteManager()->RemoveFloatingSprite(this);
1.164 + ForceRedraw();
1.165 + }
1.166 + }
1.167 + }
1.168 +
1.169 +void CWsSpriteBase::CheckSizesL()
1.170 + {
1.171 + iMaxSize.SetSize(0,0);
1.172 + for(TInt index=0;index<iMembers->Count();index++)
1.173 + {
1.174 + CWsSpriteMember *wsm=(*iMembers)[index];
1.175 + if (wsm->iBitmap)
1.176 + {
1.177 + TSize size=wsm->iBitmap->SizeInPixels();
1.178 + if (wsm->iMaskBitmap)
1.179 + {
1.180 + TSize maskSize=wsm->iMaskBitmap->SizeInPixels();
1.181 + if (maskSize.iWidth<size.iWidth || maskSize.iHeight<size.iHeight)
1.182 + OwnerPanic(EWservPanicMaskSize);
1.183 + }
1.184 + if (size.iWidth>iMaxSize.iWidth)
1.185 + iMaxSize.iWidth=size.iWidth;
1.186 + if (size.iHeight>iMaxSize.iHeight)
1.187 + iMaxSize.iHeight=size.iHeight;
1.188 + }
1.189 + }
1.190 + }
1.191 +
1.192 +void CWsSpriteBase::ConstructL(TUint aFlags, CWsWindow *aWindow)
1.193 + {
1.194 + // Common part of construct for both sprites and pointer cursors
1.195 + iFlags=aFlags;
1.196 +/*
1.197 + if (iFlags&ESpriteNoChildClip)
1.198 + iLink.iPriority+=ENoChildPriorityBoost;
1.199 + if (iFlags&ESpritePointer)
1.200 + iLink.iPriority+=EPointerPriorityBoost;
1.201 +*/
1.202 + iWin=aWindow;
1.203 + TCallBack callback(TimerCallBack,this);
1.204 + iDeltaTimerEntry.Set(callback);
1.205 + iMembers=new(ELeave) CArrayPtrFlat<CWsSpriteMember>(10);
1.206 + }
1.207 +
1.208 +void CWsSpriteBase::AppendMemberL(const TCmdSpriteMember &aCmdSpriteMember)
1.209 + {
1.210 + CWsSpriteMember *&pwsm=iMembers->ExtendL();
1.211 + pwsm=NULL; // In case new leaves
1.212 + pwsm=new(ELeave) CWsSpriteMember();
1.213 + if (pwsm->SetL(aCmdSpriteMember))
1.214 + OwnerPanic(EWservPanicBitmap);
1.215 + }
1.216 +
1.217 +void CWsSpriteBase::CompleteL()
1.218 + {
1.219 + if (iMembers->Count() <= 0)
1.220 + {
1.221 + if(iWin)
1.222 + iWin->OwnerPanic(EWservPanicNoSpriteMember);
1.223 + //Not sure if this is a neccessary fall back if iWin is NULL.
1.224 + else if(iWin==NULL && iGroupWin)
1.225 + iGroupWin->OwnerPanic(EWservPanicNoSpriteMember);
1.226 + }
1.227 + else
1.228 + {
1.229 + iMembers->Compress();
1.230 + CheckSizesL();
1.231 + SetMember(0);
1.232 + }
1.233 + }
1.234 +
1.235 +void CWsSpriteBase::Activate()
1.236 + {
1.237 + if (iFlags&ESpriteDisabled)
1.238 + {
1.239 + iFlags&=~ESpriteDisabled;
1.240 + }
1.241 + if (iMembers->Count()>1)
1.242 + {
1.243 + QueueDeltaTimer();
1.244 + iDeltaTimer->Activate();
1.245 + }
1.246 + iFlags|=ESpriteActive;
1.247 + if(iWin)
1.248 + iWin->AddSprite(this);
1.249 + Screen()->SpriteManager()->Schedule(this);
1.250 + if(iFloating)
1.251 + {
1.252 + Screen()->SpriteManager()->AddFloatingSprite(this);
1.253 + ForceRedraw();
1.254 + }
1.255 + }
1.256 +
1.257 +void CWsSpriteBase::SetMember(TInt aIndex)
1.258 + {
1.259 + iCurIndex=aIndex;
1.260 + iPos=iBasePos+(*iMembers)[iCurIndex]->iOffset;
1.261 + if ((*iMembers)[iCurIndex]->iBitmap)
1.262 + iSize=(*iMembers)[iCurIndex]->iBitmap->SizeInPixels();
1.263 + else
1.264 + iSize.SetSize(0,0);
1.265 +
1.266 + if (iSize.iWidth > iMaxSize.iWidth || iSize.iHeight > iMaxSize.iHeight)
1.267 + {
1.268 + WS_ASSERT_DEBUG(EFalse, EWsPanicSpriteBitmapSizeChange);
1.269 + CheckSizesL();
1.270 + }
1.271 + }
1.272 +
1.273 +void CWsSpriteBase::SetPos(const TPoint &aPos)
1.274 + {
1.275 + //Non-floating anim whose window is destroyed
1.276 + if (!iFloating && iWin==NULL)
1.277 + {
1.278 + OwnerPanic(EWservPanicWindowDestroyed);
1.279 + }
1.280 +
1.281 + //Floating anim whose group window is destroyed
1.282 + if (iFloating && iGroupWin==NULL)
1.283 + {
1.284 + OwnerPanic(EWservPanicWindowDestroyed);
1.285 + }
1.286 +
1.287 + iBasePos=aPos;
1.288 + TPoint newPos(iBasePos+(*iMembers)[iCurIndex]->iOffset);
1.289 + if (iPos!=newPos)
1.290 + {
1.291 + //Ensure the region covered by the sprite before as well as after the move gets scheduled for redraw
1.292 + Screen()->SpriteManager()->Schedule(this);
1.293 + iPos=newPos;
1.294 + Screen()->SpriteManager()->Schedule(this);
1.295 + }
1.296 + }
1.297 +
1.298 +// Andy - replace delta timer with animation scheduling via screen
1.299 +void CWsSpriteBase::QueueDeltaTimer()
1.300 + {
1.301 + iDeltaTimer->Queue((*iMembers)[iCurIndex]->iInterval,iDeltaTimerEntry);
1.302 + }
1.303 +
1.304 +void CWsSpriteBase::TimerExpired()
1.305 + {
1.306 + Screen()->SpriteManager()->Schedule(this);
1.307 + SetMember((iCurIndex+1)==iMembers->Count() ? 0 : iCurIndex+1);
1.308 + Screen()->SpriteManager()->Schedule(this);
1.309 + QueueDeltaTimer();
1.310 + iScreen->Update();
1.311 + }
1.312 +
1.313 +TPoint CWsSpriteBase::Pos() const
1.314 + {
1.315 + if (iGroupWin || iWin==NULL)
1.316 + {
1.317 + return(iPos);
1.318 + }
1.319 + return(iPos+iWin->Origin());
1.320 + }
1.321 +
1.322 +TRect CWsSpriteBase::Rect() const
1.323 + {
1.324 + TRect rect;
1.325 + rect.iTl=Pos();
1.326 + rect.iBr=rect.iTl+iSize;
1.327 + return(rect);
1.328 + }
1.329 +
1.330 +void CWsSpriteBase::CommandL(TInt aOpcode, const TAny *aCmdData)
1.331 + {
1.332 + TWsSpriteCmdUnion pData;
1.333 +
1.334 + pData.any=aCmdData;
1.335 + switch(aOpcode)
1.336 + {
1.337 + case EWsSpriteOpAppendMember:
1.338 + AppendMemberL(*pData.SpriteMember);
1.339 + break;
1.340 + case EWsSpriteOpActivate:
1.341 + if(!(iFlags&ESpriteActive))
1.342 + {
1.343 + CompleteL();
1.344 + }
1.345 + break;
1.346 + case EWsSpriteOpUpdateMember:
1.347 + if (pData.UpdateMember->index==iCurIndex)
1.348 + {
1.349 + TRect rect(Pos(), iMaxSize);
1.350 + Screen()->SpriteManager()->Schedule(this,&rect);
1.351 + }
1.352 + break;
1.353 + case EWsSpriteOpUpdateMember2:
1.354 + {
1.355 + Screen()->SpriteManager()->Schedule(this);
1.356 + if (pData.UpdateMember->index<0 || pData.UpdateMember->index>=iMembers->Count())
1.357 + User::Leave(KErrArgument);
1.358 + CWsSpriteMember *member=(*iMembers)[pData.UpdateMember->index];
1.359 + TBool ret=EFalse;
1.360 + TRAPD(err,ret=UpdateMemberL(member,pData.UpdateMember->data));
1.361 + if (err==KErrNone)
1.362 + {
1.363 + TRAP(err,CheckSizesL());
1.364 + SetMember(0);
1.365 + }
1.366 + Screen()->SpriteManager()->Schedule(this);
1.367 + User::LeaveIfError(err);
1.368 + if (ret)
1.369 + OwnerPanic(EWservPanicBitmap);
1.370 + }
1.371 + break;
1.372 + default:
1.373 + OwnerPanic(EWservPanicOpcode);
1.374 + break;
1.375 + }
1.376 + }
1.377 +
1.378 +TBool CWsSpriteBase::CanBeSeen() const
1.379 + {
1.380 + if(iWin)
1.381 + return (!(iFlags&ESpriteDisabled)) && (!iWin->VisibleRegion().IsEmpty());
1.382 + else
1.383 + return (!(iFlags&ESpriteDisabled));
1.384 + }
1.385 +
1.386 +void CWsSpriteBase::Redraw(CFbsBitGc * aGc, const TRegion& aRegion)
1.387 + {
1.388 + TFlashState currentState=EFlashOn;
1.389 + if(IsFlashingEnabled())
1.390 + currentState=Screen()->SpriteManager()->CurrentSpriteFlashState(this);
1.391 +
1.392 + if(currentState==EFlashOn)
1.393 + {
1.394 + STACK_REGION region;
1.395 + region.Copy(aRegion);
1.396 + const TRegion * pr = ®ion;
1.397 + if (iClipSprite)
1.398 + {
1.399 + //PeterI iWin shouldn't be null as iClipSprite is currently only set by the text cursor (which is never floating)
1.400 + //but just in case make sure we don't derefernce if it is null.
1.401 + TPoint origin(0,0);
1.402 + if(iWin)
1.403 + origin = iWin->Origin();
1.404 + TRect rect(iBasePos + origin + iClipOffset, iClipSize);
1.405 + region.ClipRect(rect);
1.406 + }
1.407 + region.ClipRect(RootWindow()->Abs());
1.408 +
1.409 + // Only need to draw if the region being redrawn overlaps the sprite
1.410 + const TRect spriteRect(Pos(), iSize);
1.411 + STACK_REGION spriteRegion;
1.412 + spriteRegion.AddRect(spriteRect);
1.413 + region.Intersect(spriteRegion);
1.414 + spriteRegion.Close();
1.415 +
1.416 + if (pr->CheckError())
1.417 + {
1.418 + if(iWin)
1.419 + pr = &iWin->VisibleRegion();
1.420 + else
1.421 + pr = &RootWindow()->WindowArea();
1.422 + }
1.423 +
1.424 + if (!pr->IsEmpty())
1.425 + {
1.426 + CWsSpriteMember *member=(*iMembers)[iCurIndex];
1.427 + if (member->iBitmap)
1.428 + {
1.429 + aGc->SetClippingRegion(pr);
1.430 +
1.431 + // Calculate which piece (rect) of the bitmap needs to be drawn
1.432 + const TRect redrawRect = pr->BoundingRect();
1.433 + TRect bitmapRect(spriteRect); // sprite rect relative to screen
1.434 + bitmapRect.Intersection(redrawRect);
1.435 + bitmapRect.Move(-Pos()); // adjust relative to bitmap origin
1.436 +
1.437 + if (member->iMaskBitmap)
1.438 + aGc->BitBltMasked(Pos() + bitmapRect.iTl, member->iBitmap, bitmapRect, member->iMaskBitmap, member->iInvertMask);
1.439 + else
1.440 + {
1.441 + aGc->SetDrawMode(member->iDrawMode);
1.442 + aGc->BitBlt(Pos() + bitmapRect.iTl, member->iBitmap, bitmapRect);
1.443 + aGc->SetDrawMode(CGraphicsContext::EDrawModePEN);
1.444 + }
1.445 + aGc->SetClippingRegion(NULL);
1.446 + }
1.447 + }
1.448 + region.Close();
1.449 + }
1.450 + //flashing sprites need to reschedule themselves after drawing
1.451 + if(IsFlashingEnabled())
1.452 + Screen()->SpriteManager()->Schedule(this);
1.453 + }
1.454 +
1.455 +TBool CWsSpriteBase::IsActivated() const
1.456 + {
1.457 + return (iFlags&ESpriteActive);
1.458 + }
1.459 +
1.460 +//
1.461 +// CWsSprite
1.462 +//
1.463 +
1.464 +CWsSprite::CWsSprite(CWsClient *owner) : CWsSpriteBase(owner,WS_HANDLE_SPRITE)
1.465 + {
1.466 + }
1.467 +
1.468 +CWsSprite::~CWsSprite()
1.469 + {
1.470 + if (!iFloating && IsActivated() && iWin && iWin->IsVisible())
1.471 + ForceRedraw();
1.472 +
1.473 + if (iAnim)
1.474 + CWsAnim::CloseAnim(iAnim);
1.475 + }
1.476 +
1.477 +void CWsSprite::ConstructL(const TWsClCmdCreateSprite &aParams)
1.478 + {
1.479 + NewObjL();
1.480 + CWsWindowBase *win;
1.481 + WsOwner()->HandleToWindow(aParams.window,&win);
1.482 + if (win->WinType()==EWinTypeGroup)
1.483 + {
1.484 + //If a sprite is attached to a group window it is floating.
1.485 + //Floating sprite drawing is performed by the sprite manager.
1.486 + iGroupWin=(CWsWindowGroup *)win;
1.487 + win=NULL; //Floating sprites aren't associated with any particular window.
1.488 + iFloating=ETrue;
1.489 + }
1.490 + CWsSpriteBase::ConstructL(aParams.flags&ESpriteNonSystemFlags,(CWsWindow *)win);
1.491 + iBasePos=aParams.pos;
1.492 + }
1.493 +
1.494 +void CWsSprite::CompleteL()
1.495 + {
1.496 + CWsSpriteBase::CompleteL();
1.497 + Activate();
1.498 + }
1.499 +
1.500 +void CWsSprite::CommandL(TInt aOpcode, const TAny *aCmdData)
1.501 + {
1.502 + TWsSpriteCmdUnion pData;
1.503 + pData.any=aCmdData;
1.504 + switch(aOpcode)
1.505 + {
1.506 + case EWsSpriteOpSetPosition:
1.507 + SetPos(*pData.Point);
1.508 + break;
1.509 + case EWsSpriteOpFree:
1.510 + delete this;
1.511 + break;
1.512 + default:
1.513 + CWsSpriteBase::CommandL(aOpcode, aCmdData);
1.514 + break;
1.515 + }
1.516 + }
1.517 +
1.518 +/**
1.519 +@see MAnimSpriteFunctions::UpdateMember
1.520 +@param aFullUpdate Not used. Wserv2 always do full back to front rendering, so there is no distinction between changes needing aFullUpdate or not
1.521 + */
1.522 +void CWsSprite::Update(TInt aMember,TRect aRect,TBool /*aFullUpdate*/)
1.523 + {
1.524 + if (iCurIndex!=aMember)
1.525 + return;
1.526 + aRect.Move(Pos());
1.527 + aRect.Intersection(iScreen->CurrentScreenSize());
1.528 + Screen()->SpriteManager()->Schedule(this, &aRect);
1.529 + }
1.530 +
1.531 +//
1.532 +// CWsPointerCursor
1.533 +//
1.534 +
1.535 +CWsPointerCursor::CWsPointerCursor(CWsClient *owner) : CWsSpriteBase(owner,WS_HANDLE_POINTER_CURSOR)
1.536 + {
1.537 + }
1.538 +
1.539 +void CWsPointerCursor::CloseObject()
1.540 + {
1.541 + RemoveFromIndex();
1.542 + Close();
1.543 + }
1.544 +
1.545 +void CWsPointerCursor::Close()
1.546 + {
1.547 + WS_ASSERT_DEBUG(iAccessCount>0, EWsPanicPointerCursorAccessCount);
1.548 + if (--iAccessCount==0)
1.549 + delete this;
1.550 + }
1.551 +
1.552 +void CWsPointerCursor::Open()
1.553 + {
1.554 + iAccessCount++;
1.555 + }
1.556 +
1.557 +CWsPointerCursor::~CWsPointerCursor()
1.558 + {
1.559 + WS_ASSERT_DEBUG(iAccessCount==0, EWsPanicPointerCursorAccessCount);
1.560 + }
1.561 +
1.562 +void CWsPointerCursor::ConstructL(const TWsClCmdCreatePointerCursor &aParams)
1.563 + {
1.564 + NewObjL();
1.565 + CWsSpriteBase::ConstructL(ESpriteNoShadows|ESpriteNoChildClip|ESpritePointer|(aParams.flags&ESpriteNonSystemFlags),RootWindow());
1.566 + Open();
1.567 + }
1.568 +
1.569 +void CWsPointerCursor::CommandL(TInt aOpcode, const TAny *aCmdData)
1.570 + {
1.571 + switch(aOpcode)
1.572 + {
1.573 + case EWsSpriteOpFree:
1.574 + CloseObject();
1.575 + break;
1.576 + default:
1.577 + CWsSpriteBase::CommandL(aOpcode, aCmdData);
1.578 + break;
1.579 + }
1.580 + }
1.581 +
1.582 +//
1.583 +// CWsCustomTextCursor
1.584 +//
1.585 +
1.586 +CWsCustomTextCursor::CWsCustomTextCursor (CWsClient *aOwner, RWsSession::TCustomTextCursorAlignment aAlignment)
1.587 +: CWsSpriteBase(aOwner, WS_HANDLE_TEXT_CURSOR), iAlignment(aAlignment)
1.588 + {
1.589 + }
1.590 +
1.591 +CWsCustomTextCursor::~CWsCustomTextCursor()
1.592 + {
1.593 + }
1.594 +
1.595 +void CWsCustomTextCursor::ConstructL(TInt aFlags)
1.596 + {
1.597 + NewObjL();
1.598 + CWsSpriteBase::ConstructL(ESpriteNoShadows|ESpriteNoChildClip|ESpritePointer|(aFlags&ESpriteNonSystemFlags), NULL);
1.599 + }
1.600 +
1.601 +void CWsCustomTextCursor::CompleteL(CWsWindow *aWin, TBool aFlash, TBool aClipSprite, const TPoint& aClipOffset, const TSize& aClipSize)
1.602 + {
1.603 + iWin = aWin;
1.604 + iFlags = aFlash ? iFlags | ESpriteFlash : iFlags & ~ESpriteFlash;
1.605 + iClipSprite = aClipSprite;
1.606 + iClipOffset = aClipOffset;
1.607 + iClipSize = aClipSize;
1.608 + CWsSpriteBase::CompleteL();
1.609 + }
1.610 +
1.611 +// Use SetPositionNoRedraw instead of SetPos when you just want to update
1.612 +// the custom text cursor position without redrawing it
1.613 +void CWsCustomTextCursor::SetPositionNoRedraw(const TPoint& aPos)
1.614 + {
1.615 + iBasePos = aPos;
1.616 + TPoint newPos(iBasePos+(*iMembers)[iCurIndex]->iOffset);
1.617 + iPos=newPos;
1.618 + }
1.619 +
1.620 +void CWsCustomTextCursor::CommandL(TInt aOpcode, const TAny *aCmdData)
1.621 + {
1.622 + switch(aOpcode)
1.623 + {
1.624 + case EWsSpriteOpFree:
1.625 + // CWsCustomTextCursor objects are owned by the text cursor list.
1.626 + // They are not deleted when the client closes it's R class.
1.627 + RemoveFromIndex();
1.628 + break;
1.629 + default:
1.630 + CWsSpriteBase::CommandL(aOpcode, aCmdData);
1.631 + break;
1.632 + }
1.633 + }
1.634 +
1.635 +//
1.636 +// CWsDeltaTimer, nicked from CDeltaTimer and tweaked so it doesn't re-activate //
1.637 +// the timers until RunL has finished running all ready timers. //
1.638 +// //
1.639 +// This is to stop a problem in Wserv where sprites could hog 100% CPU if the time //
1.640 +// it took to process them was longer than the time of the timer queued when the first //
1.641 +// sprite was updated //
1.642 +//
1.643 +
1.644 +CWsDeltaTimer* CWsDeltaTimer::NewL(TInt aPriority)
1.645 + {
1.646 + CWsDeltaTimer* wsdt=new(ELeave) CWsDeltaTimer(aPriority);
1.647 + CleanupStack::PushL(wsdt);
1.648 + User::LeaveIfError(wsdt->iTimer.CreateLocal());
1.649 + CActiveScheduler::Add(wsdt);
1.650 + CleanupStack::Pop(wsdt);
1.651 + return(wsdt);
1.652 + }
1.653 +
1.654 +CWsDeltaTimer::CWsDeltaTimer(TInt aPriority) : CActive(aPriority),iQueue(_FOFF(TWsDeltaTimerEntry,iLink))
1.655 + {
1.656 + }
1.657 +
1.658 +CWsDeltaTimer::~CWsDeltaTimer()
1.659 + {
1.660 + Cancel();
1.661 + while(iQueue.RemoveFirst()!=NULL)
1.662 + {}
1.663 + }
1.664 +
1.665 +void CWsDeltaTimer::Queue(TTimeIntervalMicroSeconds32 aTimeInMicroSeconds,TWsDeltaTimerEntry& anEntry)
1.666 + {
1.667 + TInt intervals=aTimeInMicroSeconds.Int()/CWsDeltaTimerGranularity;
1.668 + if (intervals==0)
1.669 + intervals=1;
1.670 + iQueue.Add(anEntry,intervals);
1.671 + }
1.672 +
1.673 +void CWsDeltaTimer::Activate()
1.674 + {
1.675 + // Queue a request on the timer.
1.676 + // The timer runs every tenth of a second and decremented the delta of the head of the queue.
1.677 + if (IsActive())
1.678 + return;
1.679 + if (!iQueue.IsEmpty())
1.680 + {
1.681 + SetActive();
1.682 + iTimer.After(iStatus,CWsDeltaTimerGranularity-1); // -1 to compensate for +1 in kernel!
1.683 + }
1.684 + }
1.685 +
1.686 +void CWsDeltaTimer::RunL()
1.687 + {
1.688 + // Call all zero delta callbacks
1.689 + iQueue.CountDown();
1.690 + TWsDeltaTimerEntry* ent=iQueue.RemoveFirst();
1.691 + while (ent)
1.692 + {
1.693 + ent->iCallBack.CallBack();
1.694 + ent=iQueue.RemoveFirst();
1.695 + }
1.696 + Activate();
1.697 + }
1.698 +
1.699 +void CWsDeltaTimer::DoCancel()
1.700 + {
1.701 + iTimer.Cancel();
1.702 + }
1.703 +
1.704 +void CWsDeltaTimer::Remove(TWsDeltaTimerEntry& anEntry)
1.705 + {
1.706 + if (anEntry.IsPending())
1.707 + {
1.708 + iQueue.Remove(anEntry);
1.709 + Activate();
1.710 + }
1.711 + }
1.712 +
1.713 +
1.714 +//
1.715 +// CWsSpriteManager -handles floating and flashing sprites including flashing custom text cursors. i.e. cursors
1.716 +// that have an associated sprite.
1.717 +
1.718 +
1.719 +CWsSpriteManager::CWsSpriteManager()
1.720 + {
1.721 + }
1.722 +
1.723 +CWsSpriteManager::~CWsSpriteManager()
1.724 + {
1.725 + iFloatingSprites.ResetAndDestroy();
1.726 + }
1.727 +
1.728 +CWsSpriteManager* CWsSpriteManager::NewL()
1.729 + {
1.730 + CWsSpriteManager* self = new (ELeave) CWsSpriteManager();
1.731 + CleanupStack::PushL(self);
1.732 + self->ConstructL();
1.733 + CleanupStack::Pop(self);
1.734 + return self;
1.735 + }
1.736 +
1.737 +void CWsSpriteManager::ConstructL()
1.738 + {
1.739 + }
1.740 +
1.741 +void CWsSpriteManager::AddFloatingSprite(CWsSpriteBase* aSprite)
1.742 + {
1.743 + iFloatingSprites.Append(aSprite);
1.744 + }
1.745 +
1.746 +void CWsSpriteManager::RemoveFloatingSprite(CWsSpriteBase* aSprite)
1.747 + {
1.748 + for (TInt i=0 ; i<iFloatingSprites.Count() ; i++)
1.749 + {
1.750 + if(iFloatingSprites[i]==aSprite)
1.751 + {
1.752 + //Just remove the sprite don't delete it. the manager doesn't have ownership
1.753 + iFloatingSprites.Remove(i);
1.754 + break;
1.755 + }
1.756 + }
1.757 + }
1.758 +
1.759 +void CWsSpriteManager::DrawFloatingSprites(CFbsBitGc* aGc,const TRegion& aRegion)
1.760 + {
1.761 + for (TInt i=0 ; i<iFloatingSprites.Count() ; i++)
1.762 + {
1.763 + aGc->Reset();
1.764 + iFloatingSprites[i]->Redraw(aGc, aRegion);
1.765 + }
1.766 + }
1.767 +
1.768 +void CWsSpriteManager::Schedule(CWsSpriteBase* aSprite, TRect* aRect)
1.769 + {
1.770 + if (aRect != NULL && aRect->IsEmpty())
1.771 + return;
1.772 +
1.773 + TRect rect = aSprite->Rect();
1.774 + if (aRect)
1.775 + rect.Intersection(*aRect);
1.776 +
1.777 + if(aSprite->IsFlashingEnabled())
1.778 + {
1.779 + aSprite->Screen()->ScheduleAnimation(rect,NextSpriteFlashStateChange(aSprite),0,0);
1.780 + }
1.781 + else
1.782 + {
1.783 + //Scheduling an animation "now" means it will take place at next animation which might
1.784 + //be the full animation grace period into the future (see KAnimationGrace in server.cpp)
1.785 + aSprite->Screen()->ScheduleAnimation(rect,0,0,0);
1.786 + }
1.787 + }
1.788 +
1.789 +// Sprite flashing is clamped to half second intervals in relation to the global time.
1.790 +// For the first half of each second all sprites have the EFlashOn state (visible)
1.791 +// For the second half of each second all sprites have the EFlashOff state (not visible)
1.792 +TTimeIntervalMicroSeconds CWsSpriteManager::NextSpriteFlashStateChange(const CWsSpriteBase* aSprite) const
1.793 + {
1.794 + const TTimeIntervalMicroSeconds remainder = aSprite->Screen()->Now().DateTime().MicroSecond();
1.795 + return CalculateTimeToNextFlash(remainder);
1.796 + }
1.797 +
1.798 +TTimeIntervalMicroSeconds CWsSpriteManager::NextCursorFlashStateChange() const
1.799 + {
1.800 + const TTimeIntervalMicroSeconds remainder = CWsTop::CurrentFocusScreen()->Now().DateTime().MicroSecond();
1.801 + return CalculateTimeToNextFlash(remainder);
1.802 + }
1.803 +
1.804 +TTimeIntervalMicroSeconds CWsSpriteManager::CalculateTimeToNextFlash(TTimeIntervalMicroSeconds aTime) const
1.805 + {
1.806 + TInt64 nextStateChange;
1.807 + if(aTime<KFlashHalfSecond)
1.808 + nextStateChange=KFlashHalfSecond-aTime.Int64();
1.809 + else
1.810 + nextStateChange=KFlashHalfSecond - (aTime.Int64() - KFlashHalfSecond);
1.811 + return TTimeIntervalMicroSeconds(nextStateChange);
1.812 + }
1.813 +
1.814 +TFlashState CWsSpriteManager::CurrentSpriteFlashState(const CWsSpriteBase* aSprite) const
1.815 + {
1.816 + return (aSprite->Screen()->Now().DateTime().MicroSecond()<KFlashHalfSecond)?EFlashOn:EFlashOff;
1.817 + }
1.818 +
1.819 +TFlashState CWsSpriteManager::CurrentCursorFlashState() const
1.820 + {
1.821 + return (CWsTop::CurrentFocusScreen()->Now().DateTime().MicroSecond()<KFlashHalfSecond)?EFlashOn:EFlashOff;
1.822 + }
1.823 +
1.824 +void CWsSpriteManager::CalcFloatingSpriteRgn( TRegion& aResultRgn, const TRect& aDefaultRect )
1.825 + {
1.826 + aResultRgn.Clear();
1.827 + for (TInt i=0 ; i<iFloatingSprites.Count() && !aResultRgn.CheckError(); i++)
1.828 + {
1.829 + CWsSpriteBase* sprite = iFloatingSprites[i];
1.830 + if ( sprite->CanBeSeen() && ( sprite->IsActive() || sprite->IsActivated() ) )
1.831 + {
1.832 + aResultRgn.AddRect( sprite->Rect() );
1.833 + }
1.834 + }
1.835 + aResultRgn.Tidy();
1.836 + if ( aResultRgn.CheckError() && iFloatingSprites.Count() > 0 )
1.837 + {
1.838 + aResultRgn.Clear();
1.839 + aResultRgn.AddRect( aDefaultRect );
1.840 + }
1.841 + }