os/graphics/windowing/windowserver/stdgraphic/BITMAPANIMATIONGRAPHICDRAWER.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 1995-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#include "stdgraphicdrawer.h"
sl@0
    17
#include "wsgraphicdrawercontext.h"
sl@0
    18
#include "graphics/WSGRAPHICMSGBUF.H"
sl@0
    19
#include "graphics/W32STDGRAPHICTEST.H"
sl@0
    20
#include <s32mem.h>
sl@0
    21
#include <fbs.h>
sl@0
    22
#include "W32STDGRAPHIC.H"
sl@0
    23
sl@0
    24
// CWsGraphicDrawerBitmapAnimation::CFrame \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\ 
sl@0
    25
sl@0
    26
NONSHARABLE_STRUCT(CWsGraphicDrawerBitmapAnimation::CFrame): public CBase
sl@0
    27
	{
sl@0
    28
	~CFrame();
sl@0
    29
	TFrameInfo iFrameInfo;
sl@0
    30
	CFbsBitmap* iBitmap;
sl@0
    31
	CFbsBitmap* iMask;
sl@0
    32
	mutable RRegionBuf<12> iVisibleRegion;
sl@0
    33
	};
sl@0
    34
sl@0
    35
CWsGraphicDrawerBitmapAnimation::CFrame::~CFrame()
sl@0
    36
	{
sl@0
    37
	delete iBitmap;
sl@0
    38
	delete iMask;
sl@0
    39
	iVisibleRegion.Close();
sl@0
    40
	}
sl@0
    41
sl@0
    42
// CWsGraphicDrawerBitmapAnimation \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\ 
sl@0
    43
sl@0
    44
CWsGraphicDrawerBitmapAnimation* CWsGraphicDrawerBitmapAnimation::CreateL()
sl@0
    45
	{
sl@0
    46
	return new(ELeave) CWsGraphicDrawerBitmapAnimation;
sl@0
    47
	}
sl@0
    48
sl@0
    49
CWsGraphicDrawerBitmapAnimation::CWsGraphicDrawerBitmapAnimation()
sl@0
    50
	{
sl@0
    51
	}
sl@0
    52
sl@0
    53
CWsGraphicDrawerBitmapAnimation::~CWsGraphicDrawerBitmapAnimation()
sl@0
    54
	{
sl@0
    55
	if (iContext)
sl@0
    56
		{
sl@0
    57
		iContext->Destroy();
sl@0
    58
		iContext = NULL;
sl@0
    59
		}
sl@0
    60
	iFrames.ResetAndDestroy();
sl@0
    61
	}
sl@0
    62
sl@0
    63
void CWsGraphicDrawerBitmapAnimation::ConstructL(MWsGraphicDrawerEnvironment& aEnv,const TGraphicDrawerId& aId,MWsClient& aOwner,const TDesC8& aData)
sl@0
    64
	{
sl@0
    65
	__ASSERT_COMPILE(sizeof(TInt) == sizeof(TInt32));
sl@0
    66
	RDesReadStream in(aData);
sl@0
    67
	in.PushL();
sl@0
    68
	const TInt count = in.ReadInt32L();
sl@0
    69
	for(TInt i=0; i<count; i++)
sl@0
    70
		{
sl@0
    71
		CFrame* frame = new(ELeave) CFrame;
sl@0
    72
		CleanupStack::PushL(frame);
sl@0
    73
		in.ReadL(reinterpret_cast<TUint8*>(&(frame->iFrameInfo)),sizeof(TFrameInfo));
sl@0
    74
		const TInt bitmapHandle = in.ReadInt32L();
sl@0
    75
		if(bitmapHandle)
sl@0
    76
			{
sl@0
    77
			frame->iBitmap = new(ELeave) CFbsBitmap;
sl@0
    78
			User::LeaveIfError(frame->iBitmap->Duplicate(bitmapHandle));
sl@0
    79
			}
sl@0
    80
		const TInt maskHandle = in.ReadInt32L();
sl@0
    81
		if(maskHandle)
sl@0
    82
			{
sl@0
    83
			frame->iMask = new(ELeave) CFbsBitmap;
sl@0
    84
			User::LeaveIfError(frame->iMask->Duplicate(maskHandle));
sl@0
    85
			}
sl@0
    86
		iFrames.AppendL(frame);
sl@0
    87
		CleanupStack::Pop(frame);
sl@0
    88
		TInt64 delay = frame->iFrameInfo.iDelay.Int64();
sl@0
    89
		if((delay < 0) || (delay > KMaxTUint32))
sl@0
    90
			{
sl@0
    91
			User::Leave(KErrCorrupt);
sl@0
    92
			}
sl@0
    93
		iAnimationLength += delay;
sl@0
    94
		}
sl@0
    95
	in.Pop();
sl@0
    96
	BaseConstructL(aEnv,aId,aOwner);
sl@0
    97
	if (!(aEnv.Screen(0)->ResolveObjectInterface(KMWsCompositionContext) || aEnv.Screen(0)->ResolveObjectInterface(KMWsScene)))
sl@0
    98
		{
sl@0
    99
		iContext = CWsGraphicDrawerNonNgaContext::NewL();
sl@0
   100
		}
sl@0
   101
	else
sl@0
   102
		{
sl@0
   103
		iContext = CWsGraphicDrawerNgaContext::NewL();
sl@0
   104
		}
sl@0
   105
	}
sl@0
   106
sl@0
   107
void CWsGraphicDrawerBitmapAnimation::DoDraw(MWsGc& aGc,const TRect& aRect,const TDesC8& aData) const
sl@0
   108
	{
sl@0
   109
	const TInt count = iFrames.Count();
sl@0
   110
	if(0 >= count)
sl@0
   111
		{
sl@0
   112
		return;
sl@0
   113
		}
sl@0
   114
sl@0
   115
	TWsGraphicMsgBufParser buf(aData);
sl@0
   116
	if(KErrNone != buf.Verify())
sl@0
   117
		{
sl@0
   118
		return;
sl@0
   119
		}
sl@0
   120
			
sl@0
   121
	TWsGraphicMsgAnimation anim;
sl@0
   122
 	if (KErrNone != anim.Load(buf))
sl@0
   123
 		{
sl@0
   124
 		return;
sl@0
   125
 		}
sl@0
   126
sl@0
   127
	if (KErrNone != iContext->Push(aGc))
sl@0
   128
		{
sl@0
   129
		return;
sl@0
   130
		}
sl@0
   131
sl@0
   132
	const TInt64 now_microseconds = (iAnimationLength ? anim.AnimationTime(iContext->Now(aGc),iAnimationLength).Int64(): 0LL);
sl@0
   133
	TInt64 time_microseconds = 0LL;
sl@0
   134
sl@0
   135
	// find end frame
sl@0
   136
	TInt endFrame = 0;
sl@0
   137
	while((endFrame<count) && (time_microseconds <= now_microseconds))
sl@0
   138
		{
sl@0
   139
		// work out timing
sl@0
   140
		time_microseconds += iFrames[endFrame]->iFrameInfo.iDelay.Int64();
sl@0
   141
		endFrame++;
sl@0
   142
		}
sl@0
   143
sl@0
   144
	TBool drawError = EFalse;
sl@0
   145
	// work out visible regions
sl@0
   146
	for(TInt i = 0; (i < endFrame) && !drawError; i++)
sl@0
   147
		{
sl@0
   148
		const CFrame* frame = iFrames[i];
sl@0
   149
		const TRect frameRect(frame->iFrameInfo.iFrameCoordsInPixels);
sl@0
   150
 		frame->iVisibleRegion.Clear();
sl@0
   151
sl@0
   152
 		if((i == (endFrame - 1)) || frame->iFrameInfo.iFlags & TFrameInfo::ELeaveInPlace)	// ELeave - Enum TFrameInfo::ELeaveInPlace triggers leavescan
sl@0
   153
 			{
sl@0
   154
 			frame->iVisibleRegion.AddRect(frameRect);
sl@0
   155
 			drawError = frame->iVisibleRegion.CheckError();
sl@0
   156
 			}
sl@0
   157
 		else
sl@0
   158
   			{
sl@0
   159
   			if(frame->iFrameInfo.iFlags & TFrameInfo::ERestoreToBackground) 
sl@0
   160
 				{
sl@0
   161
 				for(TInt j = 0; j <= i; j++)
sl@0
   162
 					{
sl@0
   163
 					iFrames[j]->iVisibleRegion.SubRect(frameRect);
sl@0
   164
 					// coverity[unchecked_value]
sl@0
   165
 					drawError |= iFrames[j]->iVisibleRegion.CheckError();
sl@0
   166
 					}
sl@0
   167
 				}
sl@0
   168
 			else if(!(frame->iFrameInfo.iFlags & TFrameInfo::ERestoreToPrevious))  // if no disposal method is set, treat it as leave in place
sl@0
   169
 				{
sl@0
   170
 				frame->iVisibleRegion.AddRect(frameRect);
sl@0
   171
 				drawError = frame->iVisibleRegion.CheckError();
sl@0
   172
 				}
sl@0
   173
   			}
sl@0
   174
		}
sl@0
   175
sl@0
   176
	//draw each of the visible sub frames
sl@0
   177
	for(TInt i = 0; (i < endFrame) && !drawError; i++)
sl@0
   178
		{
sl@0
   179
		const CFrame* frame = iFrames[i];
sl@0
   180
		if(frame->iBitmap)
sl@0
   181
			{
sl@0
   182
			const TRegionFix<1> bitmapRegion(frame->iFrameInfo.iFrameCoordsInPixels);
sl@0
   183
			frame->iVisibleRegion.Intersect(bitmapRegion);
sl@0
   184
			frame->iVisibleRegion.Tidy();
sl@0
   185
			if(frame->iVisibleRegion.CheckError())
sl@0
   186
				{
sl@0
   187
				drawError = ETrue;
sl@0
   188
				break;
sl@0
   189
				}
sl@0
   190
			else
sl@0
   191
				{
sl@0
   192
				const TInt clipCount = frame->iVisibleRegion.Count();
sl@0
   193
 				for(TInt j = 0; j < clipCount; j++)
sl@0
   194
					{
sl@0
   195
 					TRect frameRect = frame->iVisibleRegion.RectangleList()[j];
sl@0
   196
 					TRect clipRect(frameRect);
sl@0
   197
 					clipRect.Move(aRect.iTl);
sl@0
   198
 					clipRect.Intersection(aRect);
sl@0
   199
sl@0
   200
 					frameRect.Move(aRect.iTl);
sl@0
   201
 					frameRect.Intersection(aRect);
sl@0
   202
 					frameRect.Move(-aRect.iTl);
sl@0
   203
 					frameRect.Move(-frame->iFrameInfo.iFrameCoordsInPixels.iTl);
sl@0
   204
sl@0
   205
					if(!clipRect.IsEmpty() && !frameRect.IsEmpty())
sl@0
   206
						{
sl@0
   207
						iContext->DrawBitmap(aGc,clipRect.iTl, frame->iBitmap, frameRect, frame->iMask);
sl@0
   208
						}
sl@0
   209
					}
sl@0
   210
				}
sl@0
   211
			}
sl@0
   212
		}
sl@0
   213
sl@0
   214
	if(0 <= buf.Find(TUid::Uid(TWsGraphicFrameRate::ETypeId)))
sl@0
   215
		{
sl@0
   216
		iContext->DrawFrameRate(aGc,aRect,iFps);
sl@0
   217
		}
sl@0
   218
	iFps.Sample();
sl@0
   219
	
sl@0
   220
	if(anim.IsPlaying(iContext->Now(aGc),iAnimationLength))
sl@0
   221
		{
sl@0
   222
		iContext->ScheduleAnimation(aGc, aRect,(time_microseconds - now_microseconds));
sl@0
   223
		}
sl@0
   224
	else if(drawError)
sl@0
   225
 		{
sl@0
   226
 		iContext->ScheduleAnimation(aGc, aRect,now_microseconds + 1000000); // retry in 1 second
sl@0
   227
 		}
sl@0
   228
 
sl@0
   229
	iContext->Pop(aGc);
sl@0
   230
	}
sl@0
   231
sl@0
   232
void CWsGraphicDrawerBitmapAnimation::HandleMessage(const TDesC8& /*aData*/)
sl@0
   233
	{
sl@0
   234
	}
sl@0
   235