Update contrib.
1 // Copyright (c) 1994-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.
15 // GC and Graphics functions
19 #include "playbackgc.h"
24 #include "backedupwindow.h"
27 #include "windowgroup.h"
30 #include "Graphics/WSGRAPHICDRAWER.H"
32 #if defined(__WINS__) && defined(_DEBUG)
33 # include "offscreenbitmap.h"
34 # define DEBUGOSB { CWsOffScreenBitmap * ofb = iWin->Screen()->OffScreenBitmap(); if (ofb) ofb->Update(); }
39 CPlaybackGc * CPlaybackGc::iSelf=NULL;
41 GLREF_C RWsRegion* InternalizeRegionL(RReadStream& aReadStream);
45 void CPlaybackGc::InitStaticsL()
47 iSelf=new(ELeave) CPlaybackGc();
51 void CPlaybackGc::DeleteStatics()
57 CPlaybackGc::CPlaybackGc()
61 void CPlaybackGc::ConstructL()
63 iSelf->iScratchBitmap=new(ELeave) CFbsBitmap();
64 iSelf->iScratchMaskBitmap=new(ELeave) CFbsBitmap();
65 iGcBuf = CBufSeg::NewL(512);
68 CPlaybackGc::~CPlaybackGc()
72 iCurrentClippingRegion = NULL;
73 delete iScratchBitmap;
74 delete iScratchMaskBitmap;
77 void CPlaybackGc::Activate(CWsClientWindow * aWin, CFbsBitGc * aGc, const TRegion * aRegion)
81 iTargetRegion = aRegion;
83 iDrawRegion = iTargetRegion;
84 iMasterOrigin = iWin->Origin();
86 iGc->SetBrushColor(iWin->BackColor());
90 void CPlaybackGc::Deactivate()
96 CancelUserClippingRegion();
99 void CPlaybackGc::CancelUserClippingRegion()
101 if (iUserDefinedClippingRegion)
103 iUserDefinedClippingRegion->Destroy();
104 iUserDefinedClippingRegion = 0;
105 iDrawRegion = iTargetRegion;
109 void CPlaybackGc::SetClippingRect(const TRect &aRect)
112 iClippingRect.Move(iOrigin);
113 iClippingRectSet=ETrue;
116 void CPlaybackGc::ResetClippingRect()
118 iClippingRectSet=EFalse;
121 void CPlaybackGc::CheckPolyData(const TAny* aDataPtr, TInt aHeaderSize, TInt aNumPoints)
124 if (CWsClient::iCurrentCommand.iOpcode>0)
126 maxDataLen=CWsClient::EndOfCommandBuffer()-static_cast<const TUint8*>(aDataPtr);
130 maxDataLen=CWsClient::iCurrentCommand.iCmdLength;
132 const TInt dataSize=aHeaderSize+aNumPoints*sizeof(TPoint);
133 if (dataSize>maxDataLen)
134 GcOwnerPanic(EWservPanicBadPolyData);
137 void CPlaybackGc::DoDrawPolygon(const TWsGcCmdDrawPolygon *aDrawPolygon)
139 CheckPolyData(aDrawPolygon, sizeof(TWsGcCmdDrawPolygon), aDrawPolygon->numPoints);
140 iGc->DrawPolygon((TPoint *)(aDrawPolygon+1),aDrawPolygon->numPoints,aDrawPolygon->fillRule);
143 void CPlaybackGc::StartSegmentedDrawPolygonL(const TWsGcCmdStartSegmentedDrawPolygon *aDrawPolygon)
145 // In case a Playback have been done before all the segment is in the RedrawStore
146 // (This allocation is deleted only thanks to the EWsGcOpDrawSegmentedPolygon opcode
147 // which arrive after all the segments)
151 iPolyPoints=(TPoint *)User::AllocL(aDrawPolygon->totalNumPoints*sizeof(TPoint));
152 iPolyPointListSize=aDrawPolygon->totalNumPoints;
155 void CPlaybackGc::SegmentedDrawPolygonData(const TWsGcCmdSegmentedDrawPolygonData *aDrawPolygon)
157 if (aDrawPolygon->index<0 || (aDrawPolygon->index + aDrawPolygon->numPoints) > iPolyPointListSize)
158 GcOwnerPanic(EWservPanicBadPolyData);
159 Mem::Copy(iPolyPoints+aDrawPolygon->index,aDrawPolygon+1,aDrawPolygon->numPoints*sizeof(TPoint));
162 void CPlaybackGc::EndSegmentedPolygon()
168 void CPlaybackGc::DoDrawPolyLine(const TWsGcCmdDrawPolyLine *aDrawPolyLine, TBool aContinued)
170 TInt numPoints=aDrawPolyLine->numPoints;
171 CheckPolyData(aDrawPolyLine, sizeof(TWsGcCmdDrawPolyLine), numPoints);
172 const TPoint *points=(TPoint *)(aDrawPolyLine+1);
176 points=&aDrawPolyLine->last;
178 if (aDrawPolyLine->more) // more to come so don't draw the end point
179 iGc->DrawPolyLineNoEndPoint(points,numPoints);
181 iGc->DrawPolyLine(points,numPoints);
184 void CPlaybackGc::GcOwnerPanic(TClientPanic aPanic)
186 iGc->SetClippingRegion(NULL);
187 iCurrentClippingRegion = NULL;
188 EndSegmentedPolygon();
189 iWin->WsOwner()->PPanic(aPanic);
192 // implementing MWsGc
194 MWsClient& CPlaybackGc::Client()
196 return *(iWin->WsOwner());
199 MWsScreen& CPlaybackGc::Screen()
201 return *(iWin->Screen());
204 TPoint CPlaybackGc::GcOrigin() const
206 return (iMasterOrigin + iOrigin);
209 const TRegion& CPlaybackGc::ClippingRegion()
211 WS_ASSERT_ALWAYS(iCurrentClippingRegion,EWsPanicDrawCommandsInvalidState);
212 return* iCurrentClippingRegion;
215 CFbsBitGc& CPlaybackGc::BitGc()
220 TInt CPlaybackGc::PushBitGcSettings()
222 // the buf format is len+data where data is written by the GC's ExternalizeL()
223 CBufBase& buf = *iGcBuf;
224 const TInt start = buf.Size();
225 RBufWriteStream out(buf,start);
226 TRAPD(err,out.WriteInt32L(0));
229 TRAP(err,iGc->ExternalizeL(out));
231 if(err) //rollback addition
233 buf.Delete(start,buf.Size()-start);
237 TRAP_IGNORE(out.CommitL();) // can't see this failing
238 TPckgBuf<TInt32> pckg(buf.Size()-sizeof(TInt32)-start);
239 buf.Write(start,pckg);
244 void CPlaybackGc::PopBitGcSettings()
246 CBufBase& buf = *iGcBuf;
251 RBufReadStream in(buf,ofs);
252 TRAPD(err,chunk = in.ReadInt32L());
255 WS_ASSERT_DEBUG(err != 0, EWsPanicWsGraphic);
258 if(ofs+sizeof(TInt32)+chunk >= buf.Size()) // the last chunk?
260 TRAP_IGNORE(iGc->InternalizeL(in));
261 buf.Delete(ofs,buf.Size()-ofs);
264 ofs += chunk + sizeof(TInt32);
268 const TTime& CPlaybackGc::Now() const
270 return iWin->Screen()->Now();
273 void CPlaybackGc::ScheduleAnimation(const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow)
275 ScheduleAnimation(aRect,aFromNow,0,0);
278 void CPlaybackGc::ScheduleAnimation(const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow,const TTimeIntervalMicroSeconds& aFreq,const TTimeIntervalMicroSeconds& aStop)
280 // convert window rect to screen rect
282 rect.Move(GcOrigin());
283 // clip rect to window extent
284 rect.Intersection(iWin->Abs());
288 iWin->Screen()->ScheduleAnimation(rect,aFromNow,aFreq,aStop);
292 void CPlaybackGc::SetGcOrigin(const TPoint& aOrigin)
294 iOrigin = aOrigin - iMasterOrigin;
298 void CPlaybackGc::RemoteReadDataAndDrawL(const CWsGraphicDrawer* aGraphic, CWsClient* aOwner, const TWsGcCmdUnion &aData)
301 HBufC8* dataBuf = NULL;
302 const TInt len = aData.WsGraphic->iDataLen;
304 if ((len >= KMaxTInt / 4) || (len < 0))
306 aOwner->PPanic(EWservPanicBuffer);
308 dataBuf = HBufC8::NewLC(len);
309 TPtr8 des = dataBuf->Des();
310 aOwner->RemoteRead(des, 0);
312 if(des.Size() != len)
314 aOwner->PPanic(EWservPanicBuffer);
317 aGraphic->Draw(*this, aData.WsGraphic->iRect, data);
318 CleanupStack::PopAndDestroy(dataBuf);
321 TPtrC CPlaybackGc::BufferTPtr(TText* aStart,TInt aLen, const TDesC8& aCmdData)
323 if ((reinterpret_cast<TUint8*>(aStart) < aCmdData.Ptr()
324 || reinterpret_cast<TUint8*>(aStart+aLen) > (aCmdData.Ptr() + aCmdData.Size()) ))
326 GcOwnerPanic(EWservPanicBufferPtr);
329 gcPtr.Set(aStart,aLen);
333 void CPlaybackGc::DoDrawCommand(TWsGcOpcodes aOpcode, const TDesC8& aCmdData, const TRegion *aRegion)
336 pData.any=aCmdData.Ptr();
340 WS_ASSERT_DEBUG(iWin,EWsPanicWindowNull);
341 if (aRegion->Count()==0)
343 iGc->SetClippingRegion(aRegion);
344 WS_ASSERT_DEBUG(!iCurrentClippingRegion, EWsPanicDrawCommandsInvalidState);
345 iCurrentClippingRegion = aRegion;
349 case EWsGcOpDrawWsGraphic:
350 case EWsGcOpDrawWsGraphicPtr:
352 TRect screenRect(pData.WsGraphic->iRect);
353 screenRect.Move(GcOrigin());
354 if(iCurrentClippingRegion->Intersects(screenRect))
356 const TInt dataLen = pData.WsGraphic->iDataLen;
358 id.iId = pData.WsGraphic->iId;
359 id.iIsUid = (pData.WsGraphic->iFlags & EWsGraphicIdUid);
360 CWsClient* owner = iWin->WsOwner();
361 const CWsGraphicDrawer* graphic = owner->WindowServer().ResolveGraphic(id);
362 if(graphic && graphic->IsSharedWith(owner->SecureId()))
364 if(aOpcode == EWsGcOpDrawWsGraphicPtr)
366 TRAPD(err, RemoteReadDataAndDrawL(graphic, owner, pData))
368 WS_PANIC_DEBUG(EWsPanicWsGraphic);
371 graphic->Draw(*this,pData.WsGraphic->iRect,CWsClient::BufferTPtr8((TUint8*)(pData.WsGraphic+1),dataLen));
373 WS_ASSERT_DEBUG(!iGcBuf->Size(),EWsPanicWsGraphic);
379 case EWsGcOpMapColorsLocal:
380 iGc->MapColors(pData.MapColorsLocal->rect, pData.MapColorsLocal->colors,pData.MapColorsLocal->numPairs,pData.MapColorsLocal->mapForwards);
382 case EWsGcOpDrawPolyLineLocalBufLen:
383 iGc->DrawPolyLine(pData.DrawPolyLineLocalBufLen->points,pData.DrawPolyLineLocalBufLen->length);
385 case EWsGcOpDrawPolyLineLocal:
386 iGc->DrawPolyLine(pData.PointList);
388 case EWsGcOpDrawPolygonLocalBufLen:
389 iGc->DrawPolygon(pData.DrawPolygonLocalBufLen->points,pData.DrawPolygonLocalBufLen->length,pData.DrawPolygonLocalBufLen->fillRule);
391 case EWsGcOpDrawPolygonLocal:
392 iGc->DrawPolygon(pData.DrawPolygonLocal->pointList,pData.DrawPolygonLocal->fillRule);
394 case EWsGcOpDrawBitmapLocal:
395 iGc->DrawBitmap(pData.BitmapLocal->pos, pData.BitmapLocal->bitmap);
397 case EWsGcOpDrawBitmap2Local:
398 iGc->DrawBitmap(pData.Bitmap2Local->rect, pData.Bitmap2Local->bitmap);
400 case EWsGcOpDrawBitmap3Local:
401 iGc->DrawBitmap(pData.Bitmap3Local->rect, pData.Bitmap3Local->bitmap, pData.Bitmap3Local->srcRect);
403 case EWsGcOpDrawBitmapMaskedLocal:
404 iGc->DrawBitmapMasked(pData.iBitmapMaskedLocal->iRect, pData.iBitmapMaskedLocal->iBitmap, pData.iBitmapMaskedLocal->iSrcRect, pData.iBitmapMaskedLocal->iMaskBitmap,pData.iBitmapMaskedLocal->iInvertMask);
406 case EWsGcOpAlphaBlendBitmapsLocal:
407 iGc->AlphaBlendBitmaps(pData.AlphaBlendBitmapsLocal->point,pData.AlphaBlendBitmapsLocal->iBitmap,
408 pData.AlphaBlendBitmapsLocal->source, pData.AlphaBlendBitmapsLocal->iAlpha,
409 pData.AlphaBlendBitmapsLocal->alphaPoint);
412 case EWsGcOpDrawText:
413 iGc->DrawText(BufferTPtr((TText *)(pData.DrawText+1),pData.DrawText->length,aCmdData),pData.DrawText->pos);
415 case EWsGcOpDrawBoxTextOptimised1:
416 iGc->DrawText(BufferTPtr((TText *)(pData.BoxTextO1+1),pData.BoxTextO1->length,aCmdData),pData.BoxTextO1->box,
417 pData.BoxTextO1->baselineOffset,CGraphicsContext::ELeft,0);
419 case EWsGcOpDrawBoxTextOptimised2:
420 iGc->DrawText(BufferTPtr((TText *)(pData.BoxTextO2+1),pData.BoxTextO2->length,aCmdData),pData.BoxTextO2->box,
421 pData.BoxTextO2->baselineOffset,pData.BoxTextO2->horiz,pData.BoxTextO2->leftMrg);
423 case EWsGcOpDrawTextPtr:
424 iGc->DrawText(*pData.DrawTextPtr->text,pData.DrawTextPtr->pos);
426 case EWsGcOpDrawTextPtr1:
427 iGc->DrawText(*pData.DrawTextPtr->text);
429 case EWsGcOpDrawBoxText:
430 iGc->DrawText(BufferTPtr((TText *)(pData.BoxText+1),pData.BoxText->length,aCmdData),pData.BoxText->box,pData.BoxText->baselineOffset,pData.BoxText->width,pData.BoxText->horiz,pData.BoxText->leftMrg);
432 case EWsGcOpDrawBoxTextPtr:
433 iGc->DrawText(*pData.DrawBoxTextPtr->text,pData.DrawBoxTextPtr->box,pData.DrawBoxTextPtr->baselineOffset,pData.DrawBoxTextPtr->width,pData.DrawBoxTextPtr->horiz,pData.DrawBoxTextPtr->leftMrg);
435 case EWsGcOpDrawBoxTextPtr1:
436 iGc->DrawText(*pData.DrawBoxTextPtr->text,pData.DrawBoxTextPtr->box);
438 case EWsGcOpDrawTextVertical:
439 iGc->DrawTextVertical(BufferTPtr((TText *)(pData.DrawTextVertical+1),pData.DrawTextVertical->length,aCmdData),pData.DrawTextVertical->pos
440 ,pData.DrawTextVertical->up);
442 case EWsGcOpDrawTextVerticalPtr:
443 iGc->DrawTextVertical(*pData.DrawTextVerticalPtr->text,pData.DrawTextVerticalPtr->pos,pData.DrawTextVerticalPtr->up);
445 case EWsGcOpDrawTextVerticalPtr1:
446 iGc->DrawTextVertical(*pData.DrawTextVerticalPtr->text,pData.DrawTextVerticalPtr->up);
448 case EWsGcOpDrawBoxTextVertical:
449 iGc->DrawTextVertical(BufferTPtr((TText *)(pData.DrawBoxTextVertical+1),pData.DrawBoxTextVertical->length,aCmdData),
450 pData.DrawBoxTextVertical->box, pData.DrawBoxTextVertical->baselineOffset,
451 pData.DrawBoxTextVertical->up,(CGraphicsContext::TTextAlign)pData.DrawBoxTextVertical->vert,pData.DrawBoxTextVertical->margin);
453 case EWsGcOpDrawBoxTextVerticalPtr:
454 iGc->DrawTextVertical(*pData.DrawBoxTextVerticalPtr->text,pData.DrawBoxTextVerticalPtr->box,pData.DrawBoxTextVerticalPtr->baselineOffset
455 ,pData.DrawBoxTextVerticalPtr->width,pData.DrawBoxTextVerticalPtr->up,pData.DrawBoxTextVerticalPtr->vert,pData.DrawBoxTextVerticalPtr->margin);
457 case EWsGcOpDrawBoxTextVerticalPtr1:
458 iGc->DrawTextVertical(*pData.DrawBoxTextVerticalPtr->text,pData.DrawBoxTextVerticalPtr->box,pData.DrawBoxTextVerticalPtr->up);
460 case EWsGcOpDrawTextLocal:
461 iGc->DrawText(*pData.DrawTextLocal->desc,pData.DrawTextLocal->pos);
463 case EWsGcOpDrawBoxTextLocal:
464 iGc->DrawText(*pData.BoxTextLocal->desc,pData.BoxTextLocal->box,pData.BoxTextLocal->baselineOffset,
465 pData.BoxTextLocal->horiz,pData.BoxTextLocal->leftMrg);
467 case EWsGcOpDrawLine:
468 iGc->DrawLine(pData.DrawLine->pnt1,pData.DrawLine->pnt2);
471 iGc->DrawLine(iLinePos,*pData.Point);
474 iGc->DrawLine(iLinePos,iLinePos+(*pData.Point));
477 iGc->Plot(*pData.Point);
482 case EWsGcOpGdiBlt2Local:
483 iGc->BitBlt(pData.GdiBlt2Local->pos,pData.GdiBlt2Local->bitmap);
485 case EWsGcOpGdiBlt3Local:
486 iGc->BitBlt(pData.GdiBlt3Local->pos,pData.GdiBlt3Local->bitmap, pData.GdiBlt3Local->rect);
488 case EWsGcOpGdiBltMaskedLocal:
489 iGc->BitBltMasked(pData.GdiBltMaskedLocal->pos,pData.GdiBltMaskedLocal->bitmap,
490 pData.GdiBltMaskedLocal->rect,pData.GdiBltMaskedLocal->maskBitmap,
491 pData.GdiBltMaskedLocal->invertMask);
493 case EWsGcOpGdiWsBlt2:
494 case EWsGcOpGdiWsBlt3:
495 case EWsGcOpGdiWsBltMasked:
496 case EWsGcOpGdiWsAlphaBlendBitmaps:
497 case EWsGcOpWsDrawBitmapMasked:
499 // Andy - we continually duplicate bitmaps in here, and yet we already have them
500 // somewhere as pointers so can't we both simplify and optimize this?
501 CFbsBitmap* scratchBitmap=iScratchBitmap;
502 CFbsBitmap* scratchMaskBimap=iScratchMaskBitmap;
504 TInt handle=WsBitmapHandle(aOpcode,pData, maskHandle);
505 CWsClient* owner=iWin->WsOwner();
508 TInt wsBmpErr = KErrNone;
509 DWsBitmap *bitmap=(DWsBitmap *)owner->HandleToObj(handle, WS_HANDLE_BITMAP);
511 wsBmpErr = KErrNotFound;
513 scratchBitmap=bitmap->FbsBitmap();
514 if (wsBmpErr == KErrNone)
515 if (aOpcode==EWsGcOpGdiWsBltMasked || aOpcode==EWsGcOpGdiWsAlphaBlendBitmaps || aOpcode==EWsGcOpWsDrawBitmapMasked)
517 DWsBitmap *bitmap2=(DWsBitmap *)owner->HandleToObj(maskHandle, WS_HANDLE_BITMAP);
519 wsBmpErr = KErrNotFound;
521 scratchMaskBimap=bitmap2->FbsBitmap();
523 if (wsBmpErr == KErrNone)
527 case EWsGcOpGdiWsBlt2:
528 iGc->BitBlt(pData.GdiBlt2->pos,scratchBitmap);
530 case EWsGcOpGdiWsBlt3:
531 iGc->BitBlt(pData.GdiBlt3->pos,scratchBitmap, pData.GdiBlt3->rect);
533 case EWsGcOpGdiWsBltMasked:
535 iGc->BitBltMasked(pData.GdiBltMasked->destination,scratchBitmap,
536 pData.GdiBltMasked->source, scratchMaskBimap,
537 pData.GdiBltMasked->invertMask);
540 case EWsGcOpGdiWsAlphaBlendBitmaps:
542 iGc->AlphaBlendBitmaps(pData.AlphaBlendBitmaps->point,scratchBitmap,
543 pData.AlphaBlendBitmaps->source, scratchMaskBimap,
544 pData.AlphaBlendBitmaps->alphaPoint);
547 case EWsGcOpWsDrawBitmapMasked:
549 iGc->DrawBitmapMasked(pData.iBitmapMasked->iRect,scratchBitmap,
550 pData.iBitmapMasked->iSrcRect,scratchMaskBimap,
551 pData.iBitmapMasked->iInvertMask);
561 case EWsGcOpGdiBltMasked:
562 case EWsGcOpGdiAlphaBlendBitmaps:
563 case EWsGcOpDrawBitmap:
564 case EWsGcOpDrawBitmap2:
565 case EWsGcOpDrawBitmap3:
566 case EWsGcOpDrawBitmapMasked:
569 TInt ret = iScratchBitmap->Duplicate(FbsBitmapHandle(aOpcode, pData,maskHandle));
575 iGc->BitBlt(pData.GdiBlt2->pos,iScratchBitmap);
578 iGc->BitBlt(pData.GdiBlt3->pos,iScratchBitmap, pData.GdiBlt3->rect);
580 case EWsGcOpGdiBltMasked:
582 ret = iScratchMaskBitmap->Duplicate(pData.GdiBltMasked->maskHandle);
585 iGc->BitBltMasked(pData.GdiBltMasked->destination,iScratchBitmap,
586 pData.GdiBltMasked->source, iScratchMaskBitmap,
587 pData.GdiBltMasked->invertMask);
588 iScratchMaskBitmap->Reset();
592 case EWsGcOpGdiAlphaBlendBitmaps:
594 ret = iScratchMaskBitmap->Duplicate(pData.AlphaBlendBitmaps->alphaHandle);
597 iGc->AlphaBlendBitmaps(pData.AlphaBlendBitmaps->point, iScratchBitmap,
598 pData.AlphaBlendBitmaps->source, iScratchMaskBitmap,
599 pData.AlphaBlendBitmaps->alphaPoint);
600 iScratchMaskBitmap->Reset();
604 case EWsGcOpDrawBitmap:
605 iGc->DrawBitmap(pData.Bitmap->pos, iScratchBitmap);
607 case EWsGcOpDrawBitmap2:
608 iGc->DrawBitmap(pData.Bitmap2->rect, iScratchBitmap);
610 case EWsGcOpDrawBitmap3:
611 iGc->DrawBitmap(pData.Bitmap3->rect, iScratchBitmap, pData.Bitmap3->srcRect);
613 case EWsGcOpDrawBitmapMasked:
615 ret = iScratchMaskBitmap->Duplicate(pData.iBitmapMasked->iMaskHandle);
618 iGc->DrawBitmapMasked(pData.iBitmapMasked->iRect, iScratchBitmap,
619 pData.iBitmapMasked->iSrcRect, iScratchMaskBitmap,
620 pData.iBitmapMasked->iInvertMask);
621 iScratchMaskBitmap->Reset();
626 iScratchBitmap->Reset();
630 case EWsGcOpDrawSegmentedPolygon:
631 iGc->DrawPolygon(iPolyPoints,iPolyPointListSize,pData.DrawSegmentedPolygon->fillRule);
633 case EWsGcOpDrawPolygon:
634 DoDrawPolygon(pData.Polygon);
636 case EWsGcOpDrawPolyLine:
637 DoDrawPolyLine(pData.PolyLine, EFalse);
639 case EWsGcOpDrawPolyLineContinued:
640 DoDrawPolyLine(pData.PolyLine, ETrue);
643 iGc->Clear(TRect(iWin->Size()));
645 case EWsGcOpClearRect:
646 iGc->Clear(*pData.Rect);
648 case EWsGcOpDrawRect:
649 iGc->DrawRect(*pData.Rect);
651 case EWsGcOpDrawEllipse:
652 iGc->DrawEllipse(*pData.Rect);
654 case EWsGcOpDrawRoundRect:
655 iGc->DrawRoundRect(*pData.Rect,pData.RoundRect->ellipse);
658 iGc->DrawArc(*pData.Rect,pData.ArcOrPie->start,pData.ArcOrPie->end);
661 iGc->DrawPie(*pData.Rect,pData.ArcOrPie->start,pData.ArcOrPie->end);
663 case EWsGcOpCopyRect:
664 iGc->CopyRect(pData.CopyRect->pos,*pData.Rect);
666 case EWsGcOpMapColors:
667 iGc->MapColors(pData.MapColors->rect,(TRgb *)(pData.MapColors+1),pData.MapColors->numPairs,pData.MapColors->mapForwards);
670 TRAP_IGNORE(iWin->OwnerPanic(EWservPanicOpcode));
673 // DEBUGOSB // comment in for per-draw-command debug osb updates
674 iGc->SetClippingRegion(NULL);
675 iCurrentClippingRegion = NULL;
678 TInt CPlaybackGc::WsBitmapHandle(TInt aOpcode, const TWsGcCmdUnion &pData, TInt& aMaskHandle)
683 case EWsGcOpGdiWsBlt2:
684 handle=pData.GdiBlt2->handle;
686 case EWsGcOpGdiWsBlt3:
687 handle=pData.GdiBlt3->handle;
689 case EWsGcOpGdiWsBltMasked:
690 handle=pData.GdiBltMasked->handle;
691 aMaskHandle = pData.GdiBltMasked->maskHandle;
693 case EWsGcOpGdiWsAlphaBlendBitmaps:
694 handle=pData.AlphaBlendBitmaps->bitmapHandle;
695 aMaskHandle = pData.AlphaBlendBitmaps->alphaHandle;
697 case EWsGcOpWsDrawBitmapMasked:
698 handle=pData.iBitmapMasked->iHandle;
699 aMaskHandle=pData.iBitmapMasked->iMaskHandle;
705 TInt CPlaybackGc::FbsBitmapHandle(TInt aOpcode, const TWsGcCmdUnion &pData, TInt& aMaskHandle)
711 case EWsGcOpGdiWsBlt2:
712 case EWsGcOpGdiWsBlt3:
713 case EWsGcOpGdiWsBltMasked:
714 case EWsGcOpGdiWsAlphaBlendBitmaps:
717 DWsBitmap *bitmap=(DWsBitmap *)iWin->WsOwner()->HandleToObj(WsBitmapHandle(aOpcode,pData, maskHandle), WS_HANDLE_BITMAP);
718 WS_ASSERT_DEBUG(bitmap, EWsPanicDrawCommandsInvalidState);
720 handle=bitmap->FbsBitmap()->Handle();
721 if (aOpcode==EWsGcOpGdiWsBltMasked || aOpcode==EWsGcOpGdiWsAlphaBlendBitmaps)
723 DWsBitmap *bitmap2=(DWsBitmap *)iWin->WsOwner()->HandleToObj(maskHandle, WS_HANDLE_BITMAP);
724 WS_ASSERT_DEBUG(bitmap2, EWsPanicDrawCommandsInvalidState);
726 aMaskHandle=bitmap2->FbsBitmap()->Handle();
731 handle=pData.GdiBlt2->handle;
734 handle=pData.GdiBlt3->handle;
736 case EWsGcOpGdiBltMasked:
737 handle=pData.GdiBltMasked->handle;
738 aMaskHandle=pData.GdiBltMasked->maskHandle;
740 case EWsGcOpGdiAlphaBlendBitmaps:
741 handle=pData.AlphaBlendBitmaps->bitmapHandle;
742 aMaskHandle=pData.AlphaBlendBitmaps->alphaHandle;
744 case EWsGcOpDrawBitmap:
745 handle=pData.Bitmap->handle;
747 case EWsGcOpDrawBitmap2:
748 handle=pData.Bitmap2->handle;
750 case EWsGcOpDrawBitmap3:
751 handle=pData.Bitmap3->handle;
753 case EWsGcOpDrawBitmapMasked:
754 handle=pData.iBitmapMasked->iHandle;
755 aMaskHandle=pData.iBitmapMasked->iMaskHandle;
762 void CPlaybackGc::UpdateJustification(TText* aText,TInt aLen, const TDesC8& aCmdData)
764 iGc->UpdateJustification(BufferTPtr(aText,aLen,aCmdData));
767 void CPlaybackGc::DoDrawing(TWsGcOpcodes aOpcode, const TDesC8& aCmdData)
770 pData.any=aCmdData.Ptr();
772 // Andy. We do this every time? Shouldn't this be set up for us already by the render stage?
773 // infact, aren't we breaking the render stage by doing it here?
774 iGc->SetUserDisplayMode(iWin->DisplayMode());
775 if (iClippingRectSet)
777 iGc->SetOrigin(iMasterOrigin);
778 iGc->SetClippingRect(iClippingRect);
780 iGc->SetOrigin(iMasterOrigin + iOrigin);
782 DoDrawCommand(aOpcode,aCmdData,iDrawRegion);
784 iGc->CancelClippingRect();
785 iGc->SetUserDisplayMode(ENone);
788 case EWsGcOpDrawLine:
789 iLinePos=pData.DrawLine->pnt2;
794 iLinePos=(*pData.Point);
798 iLinePos+=(*pData.Point);
800 case EWsGcOpDrawSegmentedPolygon:
801 EndSegmentedPolygon();
803 case EWsGcOpDrawText:
804 UpdateJustification((TText *)(pData.DrawText+1),pData.DrawText->length,aCmdData);
806 case EWsGcOpDrawTextVertical:
807 UpdateJustification((TText *)(pData.DrawTextVertical+1),pData.DrawTextVertical->length,aCmdData);
809 case EWsGcOpDrawBoxText:
810 UpdateJustification((TText *)(pData.BoxText+1),pData.BoxText->length,aCmdData);
812 case EWsGcOpDrawBoxTextOptimised1:
813 UpdateJustification((TText *)(pData.BoxTextO1+1),pData.BoxTextO1->length,aCmdData);
815 case EWsGcOpDrawBoxTextOptimised2:
816 UpdateJustification((TText *)(pData.BoxTextO2+1),pData.BoxTextO2->length,aCmdData);
818 case EWsGcOpDrawBoxTextVertical:
819 UpdateJustification((TText *)(pData.DrawBoxTextVertical+1),pData.DrawBoxTextVertical->length,aCmdData);
821 case EWsGcOpDrawTextLocal:
822 iGc->UpdateJustification(*pData.DrawTextLocal->desc);
824 case EWsGcOpDrawBoxTextLocal:
825 iGc->UpdateJustification(*pData.BoxTextLocal->desc);
827 case EWsGcOpDrawTextPtr:
828 iGc->UpdateJustification(*pData.DrawTextPtr->text);
830 case EWsGcOpDrawTextVerticalPtr:
831 iGc->UpdateJustification(*pData.DrawTextVerticalPtr->text);
833 case EWsGcOpDrawBoxTextPtr:
834 iGc->UpdateJustification(*pData.DrawBoxTextPtr->text);
836 case EWsGcOpDrawBoxTextVerticalPtr:
837 iGc->UpdateJustification(*pData.DrawBoxTextVerticalPtr->text);
842 void CPlaybackGc::CommandL(TWsGcOpcodes aOpcode, const TDesC8& aCmdData)
845 pData.any=aCmdData.Ptr();
849 case EWsGcOpStartSegmentedDrawPolygon:
850 StartSegmentedDrawPolygonL(pData.StartSegmentedDrawPolygon);
852 case EWsGcOpSegmentedDrawPolygonData:
853 SegmentedDrawPolygonData(pData.SegmentedDrawPolygonData);
855 case EWsGcOpSetClippingRegion:
856 WS_ASSERT_DEBUG(aOpcode != EWsGcOpSetClippingRegion, EWsPanicDrawCommandsInvalidState);
858 case EWsGcOpSetClippingRect:
859 SetClippingRect(*pData.Rect);
861 case EWsGcOpCancelClippingRect:
864 case EWsGcOpCancelClippingRegion:
865 CancelUserClippingRegion();
867 case EWsGcOpSetFaded:
868 iGc->SetFaded(*pData.Bool);
870 case EWsGcOpSetFadeParams:
871 iGc->SetFadingParameters(WservEncoding::ExtractFirst8BitValue(*pData.UInt),WservEncoding::ExtractSecond8BitValue(*pData.UInt));
873 case EWsGcOpSetDrawMode:
874 iGc->SetDrawMode((CGraphicsContext::TDrawMode)*pData.UInt);
877 if (CWsFontCache::Instance()->UseFont(iFont, *pData.UInt))
878 iGc->UseFont(*pData.UInt);
880 iGc->UseFontNoDuplicate(iFont);
882 case EWsGcOpDiscardFont:
883 CWsFontCache::Instance()->ReleaseFont(iFont);
886 case EWsGcOpSetUnderlineStyle:
887 iGc->SetUnderlineStyle(*pData.SetUnderlineStyle);
889 case EWsGcOpSetStrikethroughStyle:
890 iGc->SetStrikethroughStyle(*pData.SetStrikethroughStyle);
892 case EWsGcOpUseBrushPattern:
893 iGc->UseBrushPattern(*pData.handle);
895 case EWsGcOpDiscardBrushPattern:
896 iGc->DiscardBrushPattern();
898 case EWsGcOpSetBrushColor:
899 iGc->SetBrushColor(*pData.rgb);
901 case EWsGcOpSetPenColor:
902 iGc->SetPenColor(*pData.rgb);
904 case EWsGcOpSetPenStyle:
905 iGc->SetPenStyle((CGraphicsContext::TPenStyle)*pData.UInt);
907 case EWsGcOpSetPenSize:
908 iGc->SetPenSize(*pData.Size);
910 case EWsGcOpSetBrushStyle:
911 iGc->SetBrushStyle((CGraphicsContext::TBrushStyle)*pData.UInt);
914 CWsFontCache::Instance()->ReleaseFont(iFont);
918 iGc->SetBrushColor(iWin->BackColor());
920 case EWsGcOpSetBrushOrigin:
921 iGc->SetBrushOrigin(*pData.Point);
923 case EWsGcOpSetDitherOrigin:
924 iGc->SetDitherOrigin(*pData.Point);
926 case EWsGcOpSetWordJustification:
927 iGc->SetWordJustification(pData.SetJustification->excessWidth, pData.SetJustification->numGaps);
929 case EWsGcOpSetCharJustification:
930 iGc->SetCharJustification(pData.SetJustification->excessWidth, pData.SetJustification->numGaps);
932 case EWsGcOpSetOrigin:
933 SetOrigin(*pData.Point);
935 case EWsGcOpSetOpaque:
936 // Andy - opaque drawing has not been implemented yet.
937 //SetOpaque(*pData.Bool);
939 case EWsGcOpSetShadowColor:
940 iGc->SetShadowColor(*pData.rgb);
942 default: // Assume remaining functions will draw
944 DoDrawing(aOpcode,aCmdData);
950 void CPlaybackGc::SetOrigin(const TPoint &aOrigin)
955 /*------------------------------------------------------------------------------
956 Description: Retrieves graphics context information back from a given buffer
957 from a given start position.
958 -----------------------------------------------------------------------------*/
959 void CPlaybackGc::InternalizeL(const CBufBase& aBuffer,TInt& aStartPos)
961 // Open the stream used for the input from the given start position
963 RBufReadStream bufReadStream;
964 bufReadStream.Open(aBuffer,aStartPos);
965 CleanupClosePushL(bufReadStream);
967 // Read the font/bitmap server data
968 iGc->InternalizeL(bufReadStream);
970 iOrigin.iX = bufReadStream.ReadInt32L();
971 iOrigin.iY = bufReadStream.ReadInt32L();
973 iClippingRectSet=bufReadStream.ReadInt8L();
975 // If there is a clipping rectangle data read it.
976 if (iClippingRectSet)
977 bufReadStream>>iClippingRect;
979 // Force FbsBitGc to reset its user clipping rect in case orientation has changed.
980 // The user clipping rect of FbsBitGc will be set to iClippingRect later before
981 // drawing the command.
982 iGc->CancelClippingRect();
984 // Read the clipping region data
985 InternalizeClippingRegionL(bufReadStream);
987 // Read the Alpha values for Brush and Pen colors.
988 InternalizeAlphaValueL(bufReadStream);
990 CleanupStack::PopAndDestroy(&bufReadStream);
993 /*------------------------------------------------------------------------------
994 Description: Retrieves TRgb::alpha value information back from a given buffer
995 and updates the Brushcolor with the same.
996 -----------------------------------------------------------------------------*/
997 void CPlaybackGc::InternalizeAlphaValueL(RReadStream& aReadStream)
999 TRgb brushColor(iGc->BrushColor());
1000 brushColor.SetAlpha(aReadStream.ReadUint8L());
1001 iGc->SetBrushColor(brushColor);
1002 TRgb penColor(iGc->PenColor());
1003 penColor.SetAlpha(aReadStream.ReadUint8L());
1004 iGc->SetPenColor(penColor);
1007 /*------------------------------------------------------------------------------
1008 Description: Helper method to retrieve clipping region data from a given
1010 -----------------------------------------------------------------------------*/
1011 void CPlaybackGc::InternalizeClippingRegionL(RReadStream& aReadStream)
1013 WS_ASSERT_DEBUG(iTargetRegion, EWsPanicDrawCommandsInvalidState);
1014 // Read flag to indicate if client had defined a clipping region
1015 TBool clipRegion = aReadStream.ReadInt8L();
1016 CancelUserClippingRegion();
1019 // Note that this clipping region is in window relative coordinates when
1020 // received from the client (and being stored) but is in screen relative
1021 // coordinates after being retrieved from the redraw store.
1022 iUserDefinedClippingRegion = InternalizeRegionL(aReadStream);
1023 iUserDefinedClippingRegion->Offset(iWin->Origin());
1024 iUserDefinedClippingRegion->Intersect(*iTargetRegion);
1025 if (iUserDefinedClippingRegion->CheckError()) // fallback to no user clipping region
1027 CancelUserClippingRegion();
1031 iDrawRegion = iUserDefinedClippingRegion;
1037 This pretty much replaces the whole of what was TDrawDestination
1038 This can only be sensibly called from outside a sequence of drawing commands,
1039 since it negates any user defined clipping regions.
1041 void CPlaybackGc::SetTargetRegion(const TRegion* aRegion)
1043 iTargetRegion = aRegion;
1044 iDrawRegion = iTargetRegion;
1045 CancelUserClippingRegion();
1048 void CPlaybackGc::Reset()
1053 TAny * CPlaybackGc::ResolveObjectInterface(TUint aId)
1057 case MWsWindow::EWsObjectInterfaceId:
1058 return dynamic_cast<MWsWindow *>(iWin);
1059 case MWsFader::EWsObjectInterfaceId:
1060 return iWin->Screen()->Fader();