Update contrib.
1 // Copyright (c) 1999-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
21 #include "windowgroup.h"
25 #include "offscreenbitmap.h"
29 static const TInt64 KFlashHalfSecond(500000); //interval for flashing sprites
31 GLDEF_D CWsDeltaTimer *CWsSpriteBase::iDeltaTimer=NULL;
33 TInt TimerCallBack(TAny *aPtr)
35 ((CWsSpriteBase *)aPtr)->TimerExpired();
43 CWsSpriteMember::CWsSpriteMember()
47 CWsSpriteMember::~CWsSpriteMember()
53 TBool CWsSpriteMember::SetL(const TCmdSpriteMember &aCmdSpriteMember)
55 if (aCmdSpriteMember.iBitmap!=0) // Null member of sequence, time is only valid field in member data
57 iBitmap=new(ELeave) CFbsBitmap();
58 TInt ret = iBitmap->Duplicate(aCmdSpriteMember.iBitmap);
59 if (ret == KErrNoMemory)
66 if (aCmdSpriteMember.iMaskBitmap)
68 iMaskBitmap=new(ELeave) CFbsBitmap();
69 TInt ret = iMaskBitmap->Duplicate(aCmdSpriteMember.iMaskBitmap);
70 if (ret == KErrNoMemory)
77 iInvertMask=aCmdSpriteMember.iInvertMask;
78 iDrawMode=aCmdSpriteMember.iDrawMode;
79 iOffset=aCmdSpriteMember.iOffset;
81 iInterval=aCmdSpriteMember.iInterval;
89 TBool CWsSpriteBase::UpdateMemberL(CWsSpriteMember *aMember, const TCmdSpriteMember &aCmdSpriteMember)
91 CFbsBitmap *old=aMember->iBitmap;
92 CFbsBitmap *oldMask=aMember->iMaskBitmap;
93 aMember->iBitmap=NULL;
94 aMember->iMaskBitmap=NULL;
96 TRAPD(err,ret=aMember->SetL(aCmdSpriteMember));
100 delete aMember->iBitmap;
101 delete aMember->iMaskBitmap;
102 aMember->iBitmap=old;
103 aMember->iMaskBitmap=oldMask;
106 TRAP(err,CheckSizesL());
115 void CWsSpriteBase::InitStaticsL()
117 iDeltaTimer=CWsDeltaTimer::NewL(ESpriteAnimatePriority);
120 void CWsSpriteBase::DeleteStatics()
125 CWsSpriteBase::CWsSpriteBase(CWsClient *owner, WH_HANDLES aType) : CWsScreenObject(owner,aType,owner->Screen()), iClipSprite(EFalse)
129 CWsSpriteBase::~CWsSpriteBase()
133 //iDeltaTimer->Remove(iDeltaTimerEntry);
136 iMembers->ResetAndDestroy();
141 void CWsSpriteBase::ForceRedraw()
143 TRegionFix<1> region;
144 region.AddRect(Rect());
145 Screen()->AddRedrawRegion(region);
148 void CWsSpriteBase::Deactivate()
150 //Disconnect from the sprite list and hide the sprite
151 if (iFlags & ESpriteActive)
154 iWin->RemoveSprite(this);
155 if (iMembers && iMembers->Count()>1)
156 iDeltaTimer->Remove(iDeltaTimerEntry);
157 iFlags&=~ESpriteActive;
160 Screen()->SpriteManager()->RemoveFloatingSprite(this);
166 void CWsSpriteBase::CheckSizesL()
168 iMaxSize.SetSize(0,0);
169 for(TInt index=0;index<iMembers->Count();index++)
171 CWsSpriteMember *wsm=(*iMembers)[index];
174 TSize size=wsm->iBitmap->SizeInPixels();
175 if (wsm->iMaskBitmap)
177 TSize maskSize=wsm->iMaskBitmap->SizeInPixels();
178 if (maskSize.iWidth<size.iWidth || maskSize.iHeight<size.iHeight)
179 OwnerPanic(EWservPanicMaskSize);
181 if (size.iWidth>iMaxSize.iWidth)
182 iMaxSize.iWidth=size.iWidth;
183 if (size.iHeight>iMaxSize.iHeight)
184 iMaxSize.iHeight=size.iHeight;
189 void CWsSpriteBase::ConstructL(TUint aFlags, CWsWindow *aWindow)
191 // Common part of construct for both sprites and pointer cursors
194 if (iFlags&ESpriteNoChildClip)
195 iLink.iPriority+=ENoChildPriorityBoost;
196 if (iFlags&ESpritePointer)
197 iLink.iPriority+=EPointerPriorityBoost;
200 TCallBack callback(TimerCallBack,this);
201 iDeltaTimerEntry.Set(callback);
202 iMembers=new(ELeave) CArrayPtrFlat<CWsSpriteMember>(10);
205 void CWsSpriteBase::AppendMemberL(const TCmdSpriteMember &aCmdSpriteMember)
207 CWsSpriteMember *&pwsm=iMembers->ExtendL();
208 pwsm=NULL; // In case new leaves
209 pwsm=new(ELeave) CWsSpriteMember();
210 if (pwsm->SetL(aCmdSpriteMember))
211 OwnerPanic(EWservPanicBitmap);
214 void CWsSpriteBase::CompleteL()
216 if (iMembers->Count() <= 0)
219 iWin->OwnerPanic(EWservPanicNoSpriteMember);
220 //Not sure if this is a neccessary fall back if iWin is NULL.
221 else if(iWin==NULL && iGroupWin)
222 iGroupWin->OwnerPanic(EWservPanicNoSpriteMember);
226 iMembers->Compress();
232 void CWsSpriteBase::Activate()
234 if (iFlags&ESpriteDisabled)
236 iFlags&=~ESpriteDisabled;
238 if (iMembers->Count()>1)
241 iDeltaTimer->Activate();
243 iFlags|=ESpriteActive;
245 iWin->AddSprite(this);
246 Screen()->SpriteManager()->Schedule(this);
249 Screen()->SpriteManager()->AddFloatingSprite(this);
254 void CWsSpriteBase::SetMember(TInt aIndex)
257 iPos=iBasePos+(*iMembers)[iCurIndex]->iOffset;
258 if ((*iMembers)[iCurIndex]->iBitmap)
259 iSize=(*iMembers)[iCurIndex]->iBitmap->SizeInPixels();
263 if (iSize.iWidth > iMaxSize.iWidth || iSize.iHeight > iMaxSize.iHeight)
265 WS_ASSERT_DEBUG(EFalse, EWsPanicSpriteBitmapSizeChange);
270 void CWsSpriteBase::SetPos(const TPoint &aPos)
272 //Non-floating anim whose window is destroyed
273 if (!iFloating && iWin==NULL)
275 OwnerPanic(EWservPanicWindowDestroyed);
278 //Floating anim whose group window is destroyed
279 if (iFloating && iGroupWin==NULL)
281 OwnerPanic(EWservPanicWindowDestroyed);
285 TPoint newPos(iBasePos+(*iMembers)[iCurIndex]->iOffset);
288 //Ensure the region covered by the sprite before as well as after the move gets scheduled for redraw
289 Screen()->SpriteManager()->Schedule(this);
291 Screen()->SpriteManager()->Schedule(this);
295 // Andy - replace delta timer with animation scheduling via screen
296 void CWsSpriteBase::QueueDeltaTimer()
298 iDeltaTimer->Queue((*iMembers)[iCurIndex]->iInterval,iDeltaTimerEntry);
301 void CWsSpriteBase::TimerExpired()
303 Screen()->SpriteManager()->Schedule(this);
304 SetMember((iCurIndex+1)==iMembers->Count() ? 0 : iCurIndex+1);
305 Screen()->SpriteManager()->Schedule(this);
310 TPoint CWsSpriteBase::Pos() const
312 if (iGroupWin || iWin==NULL)
316 return(iPos+iWin->Origin());
319 TRect CWsSpriteBase::Rect() const
323 rect.iBr=rect.iTl+iSize;
327 void CWsSpriteBase::CommandL(TInt aOpcode, const TAny *aCmdData)
329 TWsSpriteCmdUnion pData;
334 case EWsSpriteOpAppendMember:
335 AppendMemberL(*pData.SpriteMember);
337 case EWsSpriteOpActivate:
338 if(!(iFlags&ESpriteActive))
343 case EWsSpriteOpUpdateMember:
344 if (pData.UpdateMember->index==iCurIndex)
346 TRect rect(Pos(), iMaxSize);
347 Screen()->SpriteManager()->Schedule(this,&rect);
350 case EWsSpriteOpUpdateMember2:
352 Screen()->SpriteManager()->Schedule(this);
353 if (pData.UpdateMember->index<0 || pData.UpdateMember->index>=iMembers->Count())
354 User::Leave(KErrArgument);
355 CWsSpriteMember *member=(*iMembers)[pData.UpdateMember->index];
357 TRAPD(err,ret=UpdateMemberL(member,pData.UpdateMember->data));
360 TRAP(err,CheckSizesL());
363 Screen()->SpriteManager()->Schedule(this);
364 User::LeaveIfError(err);
366 OwnerPanic(EWservPanicBitmap);
370 OwnerPanic(EWservPanicOpcode);
375 TBool CWsSpriteBase::CanBeSeen() const
378 return (!(iFlags&ESpriteDisabled)) && (!iWin->VisibleRegion().IsEmpty());
380 return (!(iFlags&ESpriteDisabled));
383 void CWsSpriteBase::Redraw(CFbsBitGc * aGc, const TRegion& aRegion)
385 TFlashState currentState=EFlashOn;
386 if(IsFlashingEnabled())
387 currentState=Screen()->SpriteManager()->CurrentSpriteFlashState(this);
389 if(currentState==EFlashOn)
392 region.Copy(aRegion);
393 const TRegion * pr = ®ion;
396 //PeterI iWin shouldn't be null as iClipSprite is currently only set by the text cursor (which is never floating)
397 //but just in case make sure we don't derefernce if it is null.
400 origin = iWin->Origin();
401 TRect rect(iBasePos + origin + iClipOffset, iClipSize);
402 region.ClipRect(rect);
404 region.ClipRect(RootWindow()->Abs());
406 // Only need to draw if the region being redrawn overlaps the sprite
407 const TRect spriteRect(Pos(), iSize);
408 STACK_REGION spriteRegion;
409 spriteRegion.AddRect(spriteRect);
410 region.Intersect(spriteRegion);
411 spriteRegion.Close();
413 if (pr->CheckError())
416 pr = &iWin->VisibleRegion();
418 pr = &RootWindow()->WindowArea();
423 CWsSpriteMember *member=(*iMembers)[iCurIndex];
426 aGc->SetClippingRegion(pr);
428 // Calculate which piece (rect) of the bitmap needs to be drawn
429 const TRect redrawRect = pr->BoundingRect();
430 TRect bitmapRect(spriteRect); // sprite rect relative to screen
431 bitmapRect.Intersection(redrawRect);
432 bitmapRect.Move(-Pos()); // adjust relative to bitmap origin
434 if (member->iMaskBitmap)
435 aGc->BitBltMasked(Pos() + bitmapRect.iTl, member->iBitmap, bitmapRect, member->iMaskBitmap, member->iInvertMask);
438 aGc->SetDrawMode(member->iDrawMode);
439 aGc->BitBlt(Pos() + bitmapRect.iTl, member->iBitmap, bitmapRect);
440 aGc->SetDrawMode(CGraphicsContext::EDrawModePEN);
442 aGc->SetClippingRegion(NULL);
447 //flashing sprites need to reschedule themselves after drawing
448 if(IsFlashingEnabled())
449 Screen()->SpriteManager()->Schedule(this);
452 TBool CWsSpriteBase::IsActivated() const
454 return (iFlags&ESpriteActive);
461 CWsSprite::CWsSprite(CWsClient *owner) : CWsSpriteBase(owner,WS_HANDLE_SPRITE)
465 CWsSprite::~CWsSprite()
467 if (!iFloating && IsActivated() && iWin && iWin->IsVisible())
471 CWsAnim::CloseAnim(iAnim);
474 void CWsSprite::ConstructL(const TWsClCmdCreateSprite &aParams)
478 WsOwner()->HandleToWindow(aParams.window,&win);
479 if (win->WinType()==EWinTypeGroup)
481 //If a sprite is attached to a group window it is floating.
482 //Floating sprite drawing is performed by the sprite manager.
483 iGroupWin=(CWsWindowGroup *)win;
484 win=NULL; //Floating sprites aren't associated with any particular window.
487 CWsSpriteBase::ConstructL(aParams.flags&ESpriteNonSystemFlags,(CWsWindow *)win);
488 iBasePos=aParams.pos;
491 void CWsSprite::CompleteL()
493 CWsSpriteBase::CompleteL();
497 void CWsSprite::CommandL(TInt aOpcode, const TAny *aCmdData)
499 TWsSpriteCmdUnion pData;
503 case EWsSpriteOpSetPosition:
504 SetPos(*pData.Point);
506 case EWsSpriteOpFree:
510 CWsSpriteBase::CommandL(aOpcode, aCmdData);
516 @see MAnimSpriteFunctions::UpdateMember
517 @param aFullUpdate Not used. Wserv2 always do full back to front rendering, so there is no distinction between changes needing aFullUpdate or not
519 void CWsSprite::Update(TInt aMember,TRect aRect,TBool /*aFullUpdate*/)
521 if (iCurIndex!=aMember)
524 aRect.Intersection(iScreen->CurrentScreenSize());
525 Screen()->SpriteManager()->Schedule(this, &aRect);
532 CWsPointerCursor::CWsPointerCursor(CWsClient *owner) : CWsSpriteBase(owner,WS_HANDLE_POINTER_CURSOR)
536 void CWsPointerCursor::CloseObject()
542 void CWsPointerCursor::Close()
544 WS_ASSERT_DEBUG(iAccessCount>0, EWsPanicPointerCursorAccessCount);
545 if (--iAccessCount==0)
549 void CWsPointerCursor::Open()
554 CWsPointerCursor::~CWsPointerCursor()
556 WS_ASSERT_DEBUG(iAccessCount==0, EWsPanicPointerCursorAccessCount);
559 void CWsPointerCursor::ConstructL(const TWsClCmdCreatePointerCursor &aParams)
562 CWsSpriteBase::ConstructL(ESpriteNoShadows|ESpriteNoChildClip|ESpritePointer|(aParams.flags&ESpriteNonSystemFlags),RootWindow());
566 void CWsPointerCursor::CommandL(TInt aOpcode, const TAny *aCmdData)
570 case EWsSpriteOpFree:
574 CWsSpriteBase::CommandL(aOpcode, aCmdData);
580 // CWsCustomTextCursor
583 CWsCustomTextCursor::CWsCustomTextCursor (CWsClient *aOwner, RWsSession::TCustomTextCursorAlignment aAlignment)
584 : CWsSpriteBase(aOwner, WS_HANDLE_TEXT_CURSOR), iAlignment(aAlignment)
588 CWsCustomTextCursor::~CWsCustomTextCursor()
592 void CWsCustomTextCursor::ConstructL(TInt aFlags)
595 CWsSpriteBase::ConstructL(ESpriteNoShadows|ESpriteNoChildClip|ESpritePointer|(aFlags&ESpriteNonSystemFlags), NULL);
598 void CWsCustomTextCursor::CompleteL(CWsWindow *aWin, TBool aFlash, TBool aClipSprite, const TPoint& aClipOffset, const TSize& aClipSize)
601 iFlags = aFlash ? iFlags | ESpriteFlash : iFlags & ~ESpriteFlash;
602 iClipSprite = aClipSprite;
603 iClipOffset = aClipOffset;
604 iClipSize = aClipSize;
605 CWsSpriteBase::CompleteL();
608 // Use SetPositionNoRedraw instead of SetPos when you just want to update
609 // the custom text cursor position without redrawing it
610 void CWsCustomTextCursor::SetPositionNoRedraw(const TPoint& aPos)
613 TPoint newPos(iBasePos+(*iMembers)[iCurIndex]->iOffset);
617 void CWsCustomTextCursor::CommandL(TInt aOpcode, const TAny *aCmdData)
621 case EWsSpriteOpFree:
622 // CWsCustomTextCursor objects are owned by the text cursor list.
623 // They are not deleted when the client closes it's R class.
627 CWsSpriteBase::CommandL(aOpcode, aCmdData);
633 // CWsDeltaTimer, nicked from CDeltaTimer and tweaked so it doesn't re-activate //
634 // the timers until RunL has finished running all ready timers. //
636 // This is to stop a problem in Wserv where sprites could hog 100% CPU if the time //
637 // it took to process them was longer than the time of the timer queued when the first //
638 // sprite was updated //
641 CWsDeltaTimer* CWsDeltaTimer::NewL(TInt aPriority)
643 CWsDeltaTimer* wsdt=new(ELeave) CWsDeltaTimer(aPriority);
644 CleanupStack::PushL(wsdt);
645 User::LeaveIfError(wsdt->iTimer.CreateLocal());
646 CActiveScheduler::Add(wsdt);
647 CleanupStack::Pop(wsdt);
651 CWsDeltaTimer::CWsDeltaTimer(TInt aPriority) : CActive(aPriority),iQueue(_FOFF(TWsDeltaTimerEntry,iLink))
655 CWsDeltaTimer::~CWsDeltaTimer()
658 while(iQueue.RemoveFirst()!=NULL)
662 void CWsDeltaTimer::Queue(TTimeIntervalMicroSeconds32 aTimeInMicroSeconds,TWsDeltaTimerEntry& anEntry)
664 TInt intervals=aTimeInMicroSeconds.Int()/CWsDeltaTimerGranularity;
667 iQueue.Add(anEntry,intervals);
670 void CWsDeltaTimer::Activate()
672 // Queue a request on the timer.
673 // The timer runs every tenth of a second and decremented the delta of the head of the queue.
676 if (!iQueue.IsEmpty())
679 iTimer.After(iStatus,CWsDeltaTimerGranularity-1); // -1 to compensate for +1 in kernel!
683 void CWsDeltaTimer::RunL()
685 // Call all zero delta callbacks
687 TWsDeltaTimerEntry* ent=iQueue.RemoveFirst();
690 ent->iCallBack.CallBack();
691 ent=iQueue.RemoveFirst();
696 void CWsDeltaTimer::DoCancel()
701 void CWsDeltaTimer::Remove(TWsDeltaTimerEntry& anEntry)
703 if (anEntry.IsPending())
705 iQueue.Remove(anEntry);
712 // CWsSpriteManager -handles floating and flashing sprites including flashing custom text cursors. i.e. cursors
713 // that have an associated sprite.
716 CWsSpriteManager::CWsSpriteManager()
720 CWsSpriteManager::~CWsSpriteManager()
722 iFloatingSprites.ResetAndDestroy();
725 CWsSpriteManager* CWsSpriteManager::NewL()
727 CWsSpriteManager* self = new (ELeave) CWsSpriteManager();
728 CleanupStack::PushL(self);
730 CleanupStack::Pop(self);
734 void CWsSpriteManager::ConstructL()
738 void CWsSpriteManager::AddFloatingSprite(CWsSpriteBase* aSprite)
740 iFloatingSprites.Append(aSprite);
743 void CWsSpriteManager::RemoveFloatingSprite(CWsSpriteBase* aSprite)
745 for (TInt i=0 ; i<iFloatingSprites.Count() ; i++)
747 if(iFloatingSprites[i]==aSprite)
749 //Just remove the sprite don't delete it. the manager doesn't have ownership
750 iFloatingSprites.Remove(i);
756 void CWsSpriteManager::DrawFloatingSprites(CFbsBitGc* aGc,const TRegion& aRegion)
758 for (TInt i=0 ; i<iFloatingSprites.Count() ; i++)
761 iFloatingSprites[i]->Redraw(aGc, aRegion);
765 void CWsSpriteManager::Schedule(CWsSpriteBase* aSprite, TRect* aRect)
767 if (aRect != NULL && aRect->IsEmpty())
770 TRect rect = aSprite->Rect();
772 rect.Intersection(*aRect);
774 if(aSprite->IsFlashingEnabled())
776 aSprite->Screen()->ScheduleAnimation(rect,NextSpriteFlashStateChange(aSprite),0,0);
780 //Scheduling an animation "now" means it will take place at next animation which might
781 //be the full animation grace period into the future (see KAnimationGrace in server.cpp)
782 aSprite->Screen()->ScheduleAnimation(rect,0,0,0);
786 // Sprite flashing is clamped to half second intervals in relation to the global time.
787 // For the first half of each second all sprites have the EFlashOn state (visible)
788 // For the second half of each second all sprites have the EFlashOff state (not visible)
789 TTimeIntervalMicroSeconds CWsSpriteManager::NextSpriteFlashStateChange(const CWsSpriteBase* aSprite) const
791 const TTimeIntervalMicroSeconds remainder = aSprite->Screen()->Now().DateTime().MicroSecond();
792 return CalculateTimeToNextFlash(remainder);
795 TTimeIntervalMicroSeconds CWsSpriteManager::NextCursorFlashStateChange() const
797 const TTimeIntervalMicroSeconds remainder = CWsTop::CurrentFocusScreen()->Now().DateTime().MicroSecond();
798 return CalculateTimeToNextFlash(remainder);
801 TTimeIntervalMicroSeconds CWsSpriteManager::CalculateTimeToNextFlash(TTimeIntervalMicroSeconds aTime) const
803 TInt64 nextStateChange;
804 if(aTime<KFlashHalfSecond)
805 nextStateChange=KFlashHalfSecond-aTime.Int64();
807 nextStateChange=KFlashHalfSecond - (aTime.Int64() - KFlashHalfSecond);
808 return TTimeIntervalMicroSeconds(nextStateChange);
811 TFlashState CWsSpriteManager::CurrentSpriteFlashState(const CWsSpriteBase* aSprite) const
813 return (aSprite->Screen()->Now().DateTime().MicroSecond()<KFlashHalfSecond)?EFlashOn:EFlashOff;
816 TFlashState CWsSpriteManager::CurrentCursorFlashState() const
818 return (CWsTop::CurrentFocusScreen()->Now().DateTime().MicroSecond()<KFlashHalfSecond)?EFlashOn:EFlashOff;
821 void CWsSpriteManager::CalcFloatingSpriteRgn( TRegion& aResultRgn, const TRect& aDefaultRect )
824 for (TInt i=0 ; i<iFloatingSprites.Count() && !aResultRgn.CheckError(); i++)
826 CWsSpriteBase* sprite = iFloatingSprites[i];
827 if ( sprite->CanBeSeen() && ( sprite->IsActive() || sprite->IsActivated() ) )
829 aResultRgn.AddRect( sprite->Rect() );
833 if ( aResultRgn.CheckError() && iFloatingSprites.Count() > 0 )
836 aResultRgn.AddRect( aDefaultRect );