Update contrib.
1 // Copyright (c) 1995-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.
14 // Window redraw code, three sorts of redrawing are supported
15 // CRedrawMsgWindow handles it via sending a redraw message to the client
19 #include "redrawmsgwindow.h"
21 #include "playbackgc.h"
33 #include <graphics/WSGRAPHICDRAWERINTERFACE.H>
34 #include <graphics/surface.h>
35 #include "windowelementset.h"
37 #include "drawresource.h"
38 #include "../debuglog/DEBUGLOG.H"
40 const TUint KDrawBufferGranularity = 240;
41 const TInt KRedrawRegionGranularity = 8;
42 const TInt KReadBufferMaxLen=0x100;
43 const TInt KRegionCompressThreshold = 6; // number of rectangles in region at which we try to compress it
45 TBool CWsRedrawMsgWindow::iAtomic = EFalse;
47 extern CDebugLogBase *wsDebugLog;
51 #define LOG_WINDOW_DRAW_COMMANDS_START(wswin,region)
52 #define LOG_WINDOW_DRAW_COMMANDS_END(wswin)
53 #define LOG_REDRAW_SEGMENT(segmentIndex,segmentType)
54 #define LOG_REDRAW_SEGMENT_REGION(region)
55 #define LOG_PLAYBACK_GC_COMMAND(opcode,data)
59 #define LOG_WINDOW_DRAW_COMMANDS_START(wswin,region) LogDrawCommandsStart(wswin,region)
60 #define LOG_WINDOW_DRAW_COMMANDS_END(wswin) LogDrawCommandsEnd(wswin)
61 #define LOG_REDRAW_SEGMENT(segmentIndex,segmentType) LogRedrawSegment(segmentIndex, segmentType)
62 #define LOG_REDRAW_SEGMENT_REGION(region) {if(wsDebugLog){ LogRegion(region);}}
63 #define LOG_PLAYBACK_GC_COMMAND(opcode,data) {if (wsDebugLog) {wsDebugLog->Command(WS_HANDLE_GC, opcode, data, NULL);}}
65 LOCAL_C void LogRedrawSegment(TUint aSegmentIndex, CWsRedrawMsgWindow::TRedrawSegmentType aSegmentType)
69 TBuf<LogTBufSize> log;
70 TTruncateOverflow overflow;
71 _LIT(KLogRedrawSegment, ">> CRedrawSegment[%d] ");
72 log.AppendFormat(KLogRedrawSegment, &overflow, aSegmentIndex);
73 _LIT(KLogRedrawSegmentPending, "Pending");
74 _LIT(KLogRedrawSegmentRedraw, "Redraw");
77 case CWsRedrawMsgWindow::ESegmentTypePendingRedraw :
78 log.AppendFormat(KLogRedrawSegmentPending, &overflow);
80 case CWsRedrawMsgWindow::ESegmentTypeRedraw :
81 log.AppendFormat(KLogRedrawSegmentRedraw, &overflow);
87 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
91 LOCAL_C void LogDrawCommandsStart(const CWsWindow* aWsWin, const TRegion* aRegion)
95 _LIT(KLogDrawCommandsStart, ">> CWsRedrawMsgWindow::DrawCommandsL() [%S][app %d] RWindow[%d]");
96 const TDesC& clientName = aWsWin->WsOwner()->Client().FullName();
97 TBuf<LogTBufSize> log;
98 TTruncateOverflow overflow;
99 log.AppendFormat(KLogDrawCommandsStart, &overflow, &clientName, aWsWin->WsOwner()->ConnectionHandle(), aWsWin->LogHandle());
100 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
105 LOCAL_C void LogDrawCommandsEnd(const CWsWindow* aWsWin)
109 _LIT(KLogDrawCommandsEnd, "<< CWsRedrawMsgWindow::DrawCommandsL() [%S][app %d] RWindow[%d]");
110 const TDesC& clientName = aWsWin->WsOwner()->Client().FullName();
111 TBuf<LogTBufSize> log;
112 TTruncateOverflow overflow;
113 log.AppendFormat(KLogDrawCommandsEnd, &overflow, &clientName, aWsWin->WsOwner()->ConnectionHandle(), aWsWin->LogHandle());
114 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
124 CWsRedrawMsgWindow::CRedrawSegment::CRedrawSegment(CWsRedrawMsgWindow& aRedraw) : iRedraw(aRedraw)
128 CWsRedrawMsgWindow::CRedrawSegment* CWsRedrawMsgWindow::CRedrawSegment::NewLC(const TRect& aRect, TRedrawSegmentType aNewRegionType, CWsRedrawMsgWindow& aRedraw)
130 CRedrawSegment* self = new (ELeave) CRedrawSegment(aRedraw);
131 CleanupStack::PushL(self);
132 self->ConstructL(aRect, aNewRegionType);
136 void CWsRedrawMsgWindow::StaticInitL()
138 // Note that the wsini.ini setting NONREDRAWAGELIMIT is obselete in
139 // Wserv2 NGA. The effective behaviour of the system is as if the
140 // value was set to zero.
142 _LIT(KAtomicRedraws, "ATOMICREDRAWS");
143 if (WsIniFile->FindVar(KAtomicRedraws) ||
144 WsIniFile->FindVar(KWSERVIniFileVarChangeTracking)) //ChangeTracking requires atomic redraws to be enabled to
145 iAtomic = ETrue; //prevent playing redraw-segemts before they are completed.
150 static TInt FindBitmapRefByHandle(const TInt* aKey, const CFbsBitmapRef& aBitmapRef)
152 return *aKey - aBitmapRef.Handle();
155 static TInt InsertBitmapRefByHandle(const CFbsBitmapRef& aFirst, const CFbsBitmapRef& aSecond)
157 return aFirst.Handle() - aSecond.Handle();
160 void CWsRedrawMsgWindow::CRedrawSegment::ReleaseFontsBitmapsDrawableSources()
162 // release Bitmap, Font and Drawable Source handles
163 TInt count = iWsBitmapArray.Count();
165 for (ii = count - 1; ii >= 0; ii--)
167 iWsBitmapArray[ii]->DecRefCount();
168 iWsBitmapArray.Remove(ii);
171 count = iWsFontArray.Count();
172 for (ii = count - 1; ii >= 0; --ii)
174 CWsFontCache::Instance()->ReleaseFont(iWsFontArray[ii]);
175 iWsFontArray.Remove(ii);
178 iRedraw.CleanupBitmapRefArray(iBitmapHandleArray);
179 iBitmapHandleArray.Close();
181 count = iWsDrawableSourceArray.Count();
182 for(ii = count - 1 ; ii >= 0; ii--)
184 iWsDrawableSourceArray[ii]->DecRefCount();
185 iWsDrawableSourceArray.Remove(ii);
189 /* Set new rectangle and region type for initial or reset redraw region
190 @leave KErrNoMemory no memory to update region details
192 void CWsRedrawMsgWindow::CRedrawSegment::ConstructL(const TRect& aRect, TRedrawSegmentType aNewRegionType)
194 iDrawCommands = CBufSeg::NewL(KDrawBufferGranularity);
195 iCreationTime.UniversalTime();
197 iRegion.AddRect(aRect);
198 if (iRegion.CheckError())
200 User::Leave(KErrNoMemory);
202 iRedrawSegmentType = aNewRegionType;
205 CWsRedrawMsgWindow::CRedrawSegment::~CRedrawSegment()
210 delete iDrawCommands;
213 // Release Font, Bitmap and Drawable Source handles, close arrays
214 ReleaseFontsBitmapsDrawableSources();
215 iWsBitmapArray.Close();
216 iWsFontArray.Close();
217 iDrawerArray.Close();
218 iWsDrawableSourceArray.Close();
222 TInt CWsRedrawMsgWindow::CRedrawSegment::SizeInBytes() const
224 TInt size = sizeof(CWsRedrawMsgWindow::CRedrawSegment);
225 size += iDrawCommands->Size();
226 size += iBitmapHandleArray.Count() * sizeof(TInt);
227 size += iWsBitmapArray.Count() * sizeof(DWsBitmap);
228 size += iWsFontArray.Count() * sizeof(CWsFbsFont);
229 size += iDrawerArray.Count() * sizeof(TGraphicDrawerId);
230 size += iWsDrawableSourceArray.Count() * sizeof(CWsDrawableSource);
234 TBool CWsRedrawMsgWindow::CRedrawSegment::AppendNonDuplicateBitmapHandleL(TInt aHandle)
236 TBool handleAppend = EFalse;
237 TInt indexBitmapHandle = iBitmapHandleArray.Find(aHandle);
238 if (indexBitmapHandle<0)
240 iBitmapHandleArray.AppendL(aHandle);
241 handleAppend = ETrue;
246 CWsRedrawMsgWindow::CWsRedrawMsgWindow(CWsWindow *aWin)
247 :CWsWindowRedraw(aWin), iBackColor(aWin->RootWindow()->DefaultBackgroundColor()), iFlags(EBackgroundClear|EStoringEntireWindow),
248 iRedrawSegments(KRedrawRegionGranularity),
254 void CWsRedrawMsgWindow::ConstructL()
256 CWsWindowRedraw::ConstructL();
257 Invalidate(&WsWin()->Rel());
258 iWsWin->WsOwner()->RedrawQueue()->ReCalcOrder();
261 CWsRedrawMsgWindow::~CWsRedrawMsgWindow()
263 WS_ASSERT_DEBUG(iMemoryLock == 0, EWsPanicMemoryLock);
264 RemoveFromRedrawQueueIfEmpty();
266 iLocalRedrawRegion.Close();
267 iRedrawSegments.ResetAndDestroy();
268 TInt count = iFbsBitmapRefArray.Count();
269 WS_ASSERT_DEBUG(count==0,EWsPanicBitmapArrayNotEmpty);
270 iFbsBitmapRefArray.ResetAndDestroy();
274 These three functions actually check for a value they have already asserted on. This is intentional.
276 void CWsRedrawMsgWindow::ExpandCommandBufferL(TInt aLength)
278 WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
283 if (iCurrentSegment->iCurrentCommandBufferWritePos + aLength > iCurrentSegment->iDrawCommands->Size())
285 iCurrentSegment->iDrawCommands->ResizeL(iCurrentSegment->iCurrentCommandBufferWritePos + aLength);
290 void CWsRedrawMsgWindow::CommandBufferWrite(const TDesC8& aDes, TInt aLength)
292 WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
293 WS_ASSERT_DEBUG(iCurrentSegment->iCurrentCommandBufferWritePos + aLength <= iCurrentSegment->iDrawCommands->Size(), EWsPanicDrawCommandsInvalidState);
296 iCurrentSegment->iDrawCommands->Write(iCurrentSegment->iCurrentCommandBufferWritePos, aDes, aLength);
297 iCurrentSegment->iCurrentCommandBufferWritePos += aLength;
301 void CWsRedrawMsgWindow::CommandBufferWrite(const TAny* aPtr,TInt aLength)
303 WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
304 WS_ASSERT_DEBUG(iCurrentSegment->iCurrentCommandBufferWritePos + aLength <= iCurrentSegment->iDrawCommands->Size(), EWsPanicDrawCommandsInvalidState);
307 iCurrentSegment->iDrawCommands->Write(iCurrentSegment->iCurrentCommandBufferWritePos, aPtr, aLength);
308 iCurrentSegment->iCurrentCommandBufferWritePos += aLength;
312 /*------------------------------------------------------------------------------
313 Description: Processes draw commands. These are received as opcodes.
314 -----------------------------------------------------------------------------*/
315 TBool CWsRedrawMsgWindow::CommandL(TInt aOpcode, TWsWinCmdUnion &aCmd)
319 case EWsWinOpEnableOSB:
322 case EWsWinOpDisableOSB:
325 case EWsWinOpSetBackgroundColor:
326 iBackColor = *aCmd.rgb;
327 iFlags |= EBackgroundClear;
329 case EWsWinOpSetNoBackgroundColor:
330 iFlags &= ~EBackgroundClear;
332 case EWsWinOpInvalidate:
333 Invalidate(aCmd.rect);
335 case EWsWinOpInvalidateFull:
338 case EWsWinOpBeginRedraw:
339 BeginRedraw(aCmd.rect);
340 ValidateRect(aCmd.rect);
342 case EWsWinOpBeginRedrawFull:
346 case EWsWinOpEndRedraw:
349 case EWsWinOpGetInvalidRegionCount:
351 SetReply(iInvalid.Count());
354 case EWsWinOpGetInvalidRegion:
356 if ((*aCmd.Int) <= 0)
357 OwnerPanic(EWservPanicInvalidRegionCount);
358 if ((!iInvalid.CheckError()) && iInvalid.Count() == (*aCmd.Int))
360 CWsClient::ReplyBuf(iInvalid.RectangleList(),(*aCmd.Int) * sizeof(TRect));
367 case EWsWinOpStoreDrawCommands:
368 /* If the client asks us not to store commands, we still store the commands
369 for the region of the window which can be seen through the parent, but
370 won't attempt to obtain the entire window.
374 SetScope(EStoreEntireWindow);
378 // Clients that turn their redraw store off will still get one,
379 // but it will only attempt to store the current viewport.
380 SetScope(EStoreViewport);
383 case EWsWinOpHandleTransparencyUpdate: // deprecated
384 case EWsWinOpSetTransparencyBitmap: // deprecated
385 case EWsWinOpSetTransparencyFactor: // deprecated
386 case EWsWinOpSetTransparencyBitmapCWs: // deprecated
387 break; // do nothing.
388 case EWsWinOpIsRedrawStoreEnabled:
391 case EWsWinOpClearRedrawStore:
392 DiscardStoredCommands();
394 case EWsWinOpSetBackgroundSurface:
395 SetBackgroundSurfaceL(*aCmd.Surface);
397 case EWsWinOpSetBackgroundSurfaceConfig:
398 SetBackgroundSurfaceL(aCmd.SurfaceConfigurationAndTrigger->surfaceConfig, aCmd.SurfaceConfigurationAndTrigger->triggerRedraw, EFalse);
400 case EWsWinOpRemoveBackgroundSurface:
401 RemoveBackgroundSurface(*aCmd.Bool);
403 case EWsWinOpGetBackgroundSurfaceConfig:
405 TSurfaceConfiguration tempConfiguration = *aCmd.SurfaceConfiguration;
406 GetBackgroundSurfaceL(tempConfiguration);
407 TInt tempSize = aCmd.SurfaceConfiguration->Size();
408 if (sizeof(TSurfaceConfiguration)<tempSize)
409 tempSize = sizeof(TSurfaceConfiguration);
410 CWsClient::ReplyBuf(&tempConfiguration,tempSize);
421 void CWsRedrawMsgWindow::BeginRedraw(const TRect* aRect)
424 OwnerPanic(EWservPanicDrawCommandsInvalidState);
425 iFlags|=EBeginEndRedraw;
426 TRAPD(err,DoBeginRedrawL(aRect));
427 DiscardStoredCommandsIfError(err);
430 void CWsRedrawMsgWindow::DoBeginRedrawL(const TRect* aRect)
432 const TRect redrawRect = (aRect ? *aRect : TRect(WsWin()->Size()));
433 if (redrawRect.IsEmpty())
435 //Skip empty rects since they are not added to the region
436 iCurrentSegment = NULL;
440 CreateNewSegmentL(redrawRect, CWsRedrawMsgWindow::ESegmentTypePendingRedraw);
442 PromotePendingSegment();
446 void CWsRedrawMsgWindow::Invalidate(const TRect * aRect)
448 //The memory allocation in this function can trigger a call to ReleaseMemory(), which would
449 //recursively call this function again. This would cause a memory leak. To avoid this
450 //we call Lock() to block ReleaseMemory() from this object while executing this function.
455 iInvalid.Copy(iWsWin->WindowArea());
456 iInvalid.Offset(-iWsWin->Origin());
458 else if((!aRect->IsEmpty()) && aRect->IsNormalized())
460 iInvalid.AddRect(*aRect);
463 if (iWsWin->IsVisible())
466 iWsWin->WsOwner()->TriggerRedraw(); //wtf isn't the redrawq already scheduling itself?
472 Obtains a region from the redraw store, intersects it with the global redraw region which
473 we have been asked to draw to, and returns the intersection.
475 const TRegion * CWsRedrawMsgWindow::ReadRegion(const TInt aRegionNum)
477 // We are drawing to the global region, and we have a region in the redraw store to clip to.
478 // We want the intersection of these to actually draw to.
479 iLocalRedrawRegion.Copy(iRedrawSegments[aRegionNum]->iRegion);
480 iLocalRedrawRegion.Offset(WsWin()->Origin());
481 iLocalRedrawRegion.Intersect(*iRedrawRegion);
482 iLocalRedrawRegion.Tidy();
484 // If the resulting region is empty there is no point drawing its corresponding commands
485 if (iLocalRedrawRegion.IsEmpty())
488 return &iLocalRedrawRegion;
491 void CWsRedrawMsgWindow::SubtractRectFromSegmentArray(const TRect& aRect)
493 for (TInt regionNum = iRedrawSegments.Count() - 1; regionNum >= 0; --regionNum)
495 if (iRedrawSegments[regionNum]->iRedrawSegmentType != ESegmentTypePendingRedraw)
497 RWsRegion& region = iRedrawSegments[regionNum]->iRegion;
498 region.SubRect(aRect);
499 if (region.CheckError())
501 // Ouch. Drop the now broken segment and ask for a full redraw
502 delete iRedrawSegments[regionNum];
503 iRedrawSegments.Remove(regionNum);
508 // check if region has zero uncovered rectangles left
509 if (region.IsEmpty())
510 { // delete draw commands, release bitmaps and fonts
511 delete iRedrawSegments[regionNum];
512 iRedrawSegments.Remove(regionNum);
516 if (region.Count() > KRegionCompressThreshold)
517 { // tidy up the rectangles
524 // coverity[extend_simple_error]
527 /*------------------------------------------------------------------------------
528 Description: Clears out the command buffer if aError indicates an
529 error has occurred whilst storing commands.
530 -----------------------------------------------------------------------------*/
531 void CWsRedrawMsgWindow::DiscardStoredCommandsIfError(TInt aError)
533 if (aError != KErrNone)
535 // Discard stored commands by clearing out the command buffer
536 DiscardStoredCommands();
538 if(!(iFlags&ECurrentRedrawFailedStorage)) //first time we fail during the current redraw
540 iFlags |= ECurrentRedrawFailedStorage;
542 if(!(iFlags&EPreviousRedrawFailedStorage)) //unless the previous redraw failed as well (to avoid infinite loop)
544 Invalidate(); //request new redraw from the client
550 /*------------------------------------------------------------------------------
551 Description: If the graphics context has changed and we are currently storing
552 commands, store the data given by aCmdData.
554 -----------------------------------------------------------------------------*/
555 TBool CWsRedrawMsgWindow::DrawCommand(CWsGc* aGc,const TAny *aCmdData)
557 // Store commands, clearing out buffer if error occurs.
558 TRAPD(err,StoreDrawCommandL(aGc,aCmdData));
559 DiscardStoredCommandsIfError(err);
563 void CWsRedrawMsgWindow::GcAttributeChange(CWsGc* aGc,const TAny *aCmdData)
565 if (iLastDrawGc == aGc && iCurrentSegment)
568 if (IsWsFontOperation(CWsClient::iCurrentCommand.iOpcode))
572 TRAP(err,AddWsFontL(*pData.UInt));
576 TRAP(err,AppendCommandL(aCmdData));
578 DiscardStoredCommandsIfError(err);
582 // Retain the bitmap handle for the lifetime of the redraw store
583 // If the client destroys it, we will still have a reference to it
584 if (iCurrentSegment && CWsClient::iCurrentCommand.iOpcode == EWsGcOpUseBrushPattern)
589 TRAP(err, AddFbsBitmapsL(*pData.handle, 0));
590 DiscardStoredCommandsIfError(err);
594 void CWsRedrawMsgWindow::GcDeactivate(CWsGc* aGc)
596 if (iLastDrawGc==aGc)
600 inline TBool CWsRedrawMsgWindow::IsFbsBitmapOperation(TInt aOpCode) const
602 WS_ASSERT_DEBUG((EWsGcOpGdiBlt3==EWsGcOpGdiBlt2+1)&&(EWsGcOpGdiBltMasked==EWsGcOpGdiBlt3+1)
603 &&(EWsGcOpDrawBitmap2==EWsGcOpDrawBitmap+1)&&(EWsGcOpDrawBitmap3==EWsGcOpDrawBitmap2+1)&&(EWsGcOpDrawBitmapMasked==EWsGcOpDrawBitmap3+1),EWsPanicBitmapOpcodeInvalid);
604 return (aOpCode>=EWsGcOpGdiBlt2&&aOpCode<=EWsGcOpGdiBltMasked)||(aOpCode>=EWsGcOpDrawBitmap&&aOpCode<=EWsGcOpDrawBitmapMasked)||(aOpCode==EWsGcOpGdiAlphaBlendBitmaps);
607 inline TBool CWsRedrawMsgWindow::IsWsBitmapOperation(TInt aOpCode) const
609 WS_ASSERT_DEBUG((EWsGcOpGdiBlt3==EWsGcOpGdiBlt2+1)&&(EWsGcOpGdiBltMasked==EWsGcOpGdiBlt3+1)
610 &&(EWsGcOpDrawBitmap2==EWsGcOpDrawBitmap+1)&&(EWsGcOpDrawBitmap3==EWsGcOpDrawBitmap2+1)&&(EWsGcOpDrawBitmapMasked==EWsGcOpDrawBitmap3+1),EWsPanicBitmapOpcodeInvalid);
611 return (aOpCode>=EWsGcOpGdiWsBlt2&&aOpCode<=EWsGcOpGdiWsBltMasked)||(aOpCode==EWsGcOpGdiWsAlphaBlendBitmaps)||(aOpCode==EWsGcOpWsDrawBitmapMasked);
614 inline TBool CWsRedrawMsgWindow::IsWsFontOperation(TInt aOpCode) const
616 return aOpCode==EWsGcOpUseFont;
619 inline TBool CWsRedrawMsgWindow::IsDrawWsGraphicOperation(TInt aOpCode) const
621 return (aOpCode == EWsGcOpDrawWsGraphic) || (aOpCode == EWsGcOpDrawWsGraphicPtr);
624 inline TBool CWsRedrawMsgWindow::IsWsDrawableSourceOperation(TInt aOpCode) const
626 return (aOpCode == EWsGcOpDrawResourceToPos) || (aOpCode == EWsGcOpDrawResourceToRect) || (aOpCode == EWsGcOpDrawResourceFromRectToRect) || (aOpCode == EWsGcOpDrawResourceWithData);
629 void CWsRedrawMsgWindow::ReplaceAndAppendCommandL(TInt aOpcode,const TAny* aCmdData)
631 const TInt KCharWidthInBytes = 2; // # of bytes for a Unicode character
632 CWsClient* owner=iWsWin->WsOwner();
633 WS_ASSERT_DEBUG(owner,EWsPanicDrawCommandsNullSession);
635 // aCmdData doesn't contain data, it should be retrieved from client space using remote read
638 TUint16 newOpcode=EWsGcOpDrawText;
642 case EWsGcOpDrawTextPtr:
643 newOpcode=EWsGcOpDrawText;
644 strLen=cmd.DrawText->length;
646 case EWsGcOpDrawTextVerticalPtr:
647 newOpcode=EWsGcOpDrawTextVertical;
648 strLen=cmd.DrawTextVertical->length;
650 case EWsGcOpDrawBoxTextPtr:
651 newOpcode=EWsGcOpDrawBoxText;
652 strLen=cmd.BoxText->length;
654 case EWsGcOpDrawBoxTextVerticalPtr:
655 newOpcode=EWsGcOpDrawBoxTextVertical;
656 strLen=cmd.DrawBoxTextVertical->length;
658 case EWsGcOpDrawTextInContextPtr:
659 newOpcode=EWsGcOpDrawTextInContextPtr1;
660 strLen=cmd.DrawTextInContext->length;
662 case EWsGcOpDrawTextInContextVerticalPtr:
663 newOpcode=EWsGcOpDrawTextInContextVerticalPtr1;
664 strLen=cmd.DrawTextInContextVertical->length;
666 case EWsGcOpDrawBoxTextInContextPtr:
667 newOpcode=EWsGcOpDrawBoxTextInContextPtr1;
668 strLen=cmd.BoxTextInContext->length;
670 case EWsGcOpDrawBoxTextInContextVerticalPtr:
671 newOpcode=EWsGcOpDrawBoxTextInContextVerticalPtr1;
672 strLen=cmd.DrawBoxTextInContextVertical->length;
675 TInt strSize = strLen * KCharWidthInBytes;
676 TInt oldCmdLen=CWsClient::iCurrentCommand.iCmdLength;
677 TInt newCmdLen=sizeof(TWsCmdHeaderBase)+oldCmdLen+strSize;
679 ExpandCommandBufferL(newCmdLen);
680 // update current command to reflect the new command and data
681 CWsClient::iCurrentCommand.iOpcode=newOpcode;
682 CWsClient::iCurrentCommand.iOpcode|=EWsGcOpFlagDrawOp;
683 CWsClient::iCurrentCommand.iCmdLength=(TInt16)(oldCmdLen+strSize);
684 // write command header
685 CommandBufferWrite(&CWsClient::iCurrentCommand, sizeof(TWsCmdHeaderBase));
687 CommandBufferWrite(aCmdData, oldCmdLen);
690 TBuf<KReadBufferMaxLen> buf;
691 TInt len=KReadBufferMaxLen;
698 owner->RemoteRead(buf,bufOffset);
699 CommandBufferWrite(buf.Ptr(), len * KCharWidthInBytes);
705 /*------------------------------------------------------------------------------
706 Description: Stores drawing related commands into the command buffer
707 -----------------------------------------------------------------------------*/
708 void CWsRedrawMsgWindow::StoreDrawCommandL(CWsGc* aGc,const TAny *aCmdData)
710 TWsGcOpcodes currentOpcode = static_cast<TWsGcOpcodes>(CWsClient::iCurrentCommand.iOpcode);
714 // We've received a drawing operation outside of a BeginRedraw()/EndRedraw()
715 // bracketed block. This is known as a "non redraw drawing operation".
718 // If configured, panic on non redraw drawing operations.
719 if( CWsClient::DebugEnforceRedrawCallingConvention())
720 CWsClient::PanicCurrentClient(EWservPanicWindowBeginRedrawNotCalled);
723 // We get here if we are not configured to panic when a non-redraw drawing operation
725 // We ignore this DrawOp as WServ2 is not supporting non-redraw drawing because
726 // storing such non-redraw commands in the redraw store consumes significant
727 // amounts of memory, and reduces rendering performance.
729 // Issuing non-redraw drawing from a client Redrawer is a special case. Here
730 // an infinite loop would occur if we mark the area as invalid. Therefore
731 // we must always panic in this case.
732 if (iWsWin->WsOwner()->ClientProcessingRedrawEvent())
734 // Non-redraw drawing in the Redrawer!!
735 CWsClient::PanicCurrentClient(EWservPanicWindowBeginRedrawNotCalled);
739 // The Redrawer will eventually fix up the screen
745 // If there is no current segment then we have discarded it at some point
746 // since beginning this redraw.
750 pData.any = aCmdData;
751 if (IsFbsBitmapOperation(currentOpcode))
754 TInt handle = aGc->FbsBitmapHandle(currentOpcode, pData, maskHandle);
755 AddFbsBitmapsL(handle, maskHandle);
757 else if (IsWsBitmapOperation(currentOpcode))
760 TInt handle = aGc->WsBitmapHandle(currentOpcode, pData, maskHandle);
761 AddWsBitmapsL(handle, maskHandle);
763 else if (IsDrawWsGraphicOperation(currentOpcode))
765 TGraphicDrawerId drawerId;
766 drawerId.iId = pData.WsGraphic->iId;
767 drawerId.iIsUid = (pData.WsGraphic->iFlags & EWsGraphicIdUid);
768 iCurrentSegment->AddDrawerL(drawerId);
770 else if (IsWsDrawableSourceOperation(currentOpcode))
772 TInt handle = aGc->WsDrawableSourceHandle(currentOpcode, pData);
773 AddWsDrawableSourceL(handle);
776 // If the graphics context has changed since last time store the new graphics
777 // context attributes.
778 if (aGc != iLastDrawGc)
780 StoreAllGcAttributesL(aGc);
784 // For operation which requires remote read from client space, we must retrieve that data and store
785 // it in command buffer at server side and change opcode if necessary e.g EWsGcOpDrawTextPtr to EWsGcOpDrawText
786 // to avoid remote read during DoDrawing operation
787 if (IsRemoteReadRequired(currentOpcode))
788 ReplaceAndAppendCommandL(currentOpcode,aCmdData);
790 // Append the command data to the command buffer
791 AppendCommandL(aCmdData, EWsGcOpFlagDrawOp);
795 /*------------------------------------------------------------------------------
796 Description: Stores given drawing command data into the command buffer.
797 -----------------------------------------------------------------------------*/
798 void CWsRedrawMsgWindow::AppendCommandL(const TAny* aCmdData, const TUint16 aOpcodeFlags)
800 if (CWsClient::iCurrentCommand.iOpcode == EWsGcOpSetClippingRegion)
802 // The client is defining a clipping region
804 // make room for the header
805 ExpandCommandBufferL(sizeof(TWsCmdHeaderBase));
807 // Externalize the clipping region data from position after the header
808 RBufWriteStream bufWriteStream;
809 bufWriteStream.Open(*CurrentDrawCommandBuffer(), CurrentCommandBufferWritePos() + sizeof(TWsCmdHeaderBase));
810 CleanupClosePushL(bufWriteStream);
811 TInt dataLen = iLastDrawGc->ExternalizeClippingRegionL(bufWriteStream);
813 // Setup the clipping region data header
814 CWsClient::iCurrentCommand.iOpcode = EWsStoreClippingRegion;
815 CWsClient::iCurrentCommand.iCmdLength = REINTERPRET_CAST(TInt16&,dataLen);
817 // Store command header for clipping region data at current write position
818 CommandBufferWrite(&CWsClient::iCurrentCommand,sizeof(TWsCmdHeaderBase));
820 // Update write position for command data
821 iCurrentSegment->iCurrentCommandBufferWritePos += dataLen;
822 CleanupStack::PopAndDestroy(&bufWriteStream);
826 TUint16 opcode = CWsClient::iCurrentCommand.iOpcode;
827 CWsClient::iCurrentCommand.iOpcode |= aOpcodeFlags;
829 // ensure room in command buffer
830 ExpandCommandBufferL(sizeof(TWsCmdHeaderBase) + CWsClient::iCurrentCommand.iCmdLength);
832 // Store command header to current position
833 CommandBufferWrite(&CWsClient::iCurrentCommand, sizeof(TWsCmdHeaderBase));
835 // If there's command data (other than header), store it
836 if (CWsClient::iCurrentCommand.iCmdLength > 0)
838 CommandBufferWrite(aCmdData, CWsClient::iCurrentCommand.iCmdLength);
841 CWsClient::iCurrentCommand.iOpcode = opcode;
846 /*------------------------------------------------------------------------------
847 Description: Stores graphics context information into the command buffer
848 from the current write position.
849 -----------------------------------------------------------------------------*/
850 void CWsRedrawMsgWindow::StoreAllGcAttributesL(CWsGc* aGc)
852 // In order for the externalize below to work correctly from
853 // a non-zero position we have to create the header placeholder
854 ExpandCommandBufferL(sizeof(TWsCmdHeaderBase));
856 // Externalise GC attribute data. We do this before writing the
857 // header as we do not know the size of the data yet and it is
858 // part of the header.
859 TInt numOfBytesAdded = aGc->ExternalizeL(*CurrentDrawCommandBuffer(),
860 CurrentCommandBufferWritePos() + sizeof(TWsCmdHeaderBase));
863 TWsCmdHeaderBase cmdHeader;
864 cmdHeader.iCmdLength = (TInt16) numOfBytesAdded; // as calculated above
865 cmdHeader.iOpcode = (TInt16) EWsStoreAllGcAttributes;
867 // Store the header for the GC data into the space we created
868 CommandBufferWrite(&cmdHeader, sizeof(TWsCmdHeaderBase));
870 // Update write position for command data
871 iCurrentSegment->iCurrentCommandBufferWritePos += numOfBytesAdded;
874 LOCAL_C void CallSegmentEnd(TAny* aAnnotationObserver)
876 static_cast<MWsDrawAnnotationObserver*>(aAnnotationObserver)->SegmentRedrawEnd();
879 /*------------------------------------------------------------------------------
880 Description: Loops through the whole of the current command buffer, processing
882 -----------------------------------------------------------------------------*/
883 void CWsRedrawMsgWindow::DrawCommandsL()
885 LOG_WINDOW_DRAW_COMMANDS_START(WsWin(), iRedrawRegion);
886 WS_ASSERT_DEBUG(iMemoryLock > 0, EWsPanicMemoryLock);
887 static TBuf8<EClientBufferMaxSize> buf;
888 TInt regionCount = iRedrawSegments.Count();
890 for (TInt regionNum = 0; regionNum < regionCount; ++regionNum)
892 CRedrawSegment* segment = iRedrawSegments[regionNum];
893 LOG_REDRAW_SEGMENT(regionNum, segment->iRedrawSegmentType);
894 if (segment->iRedrawSegmentType == ESegmentTypePendingRedraw)
897 // The amount of commands we process is given by the value of the
898 // current write position rather than the size of the command buffer.
899 // Note: the write position is incremented as each command is stored and
900 // will typically be less than the buffer size.
901 const TInt length = segment->iCurrentCommandBufferWritePos;
903 // need to draw this region?
904 const TRegion * localDrawRegion = 0;
906 localDrawRegion = ReadRegion(regionNum);
909 MWsDrawAnnotationObserver* const annoObs = Screen()->DrawAnnotationObserver();
912 annoObs->SegmentRedrawStart(*localDrawRegion);
913 CleanupStack::PushL(TCleanupItem(CallSegmentEnd, annoObs));
915 CPlaybackGc::Instance()->SetTargetRegion(localDrawRegion);
916 LOG_REDRAW_SEGMENT_REGION(localDrawRegion)
918 TWsCmdHeaderBase header;
919 TInt pos = 0; // Set to first command position in buffer
920 CBufSeg* drawCmdBuffer = segment->iDrawCommands;
923 // Read the first command header. The associated opcode must always be
924 // EWsStoreAllGcAttributes as this is always the first stored item.
925 drawCmdBuffer->Read(pos,&header,sizeof(TWsCmdHeaderBase));
926 WS_ASSERT_DEBUG(header.iOpcode == EWsStoreAllGcAttributes, EWsPanicDrawCommandsBufferCorrupt);
929 // Read through remaining commands
932 // Get header of command
933 drawCmdBuffer->Read(pos, &header, sizeof(TWsCmdHeaderBase));
934 pos += sizeof(TWsCmdHeaderBase);
936 switch(header.iOpcode)
938 case EWsStoreAllGcAttributes:
940 // Header indicates command encapsulates gc data
941 CPlaybackGc::Instance()->Reset();
944 CPlaybackGc::Instance()->InternalizeL(*drawCmdBuffer,pos);
948 case EWsStoreClippingRegion:
950 // Clipping region data read in from current position via stream
951 RBufReadStream bufReadStream;
952 bufReadStream.Open(*drawCmdBuffer,pos);
953 CleanupClosePushL(bufReadStream);
954 CPlaybackGc::Instance()->InternalizeClippingRegionL(bufReadStream);
955 CleanupStack::PopAndDestroy(&bufReadStream);
960 // Another type of command. Read it.
961 CWsClient::iCurrentCommand.iCmdLength = header.iCmdLength;
962 drawCmdBuffer->Read(pos,buf,header.iCmdLength);
964 TInt opcode = header.iOpcode;
967 if (opcode & EWsGcOpFlagDrawOp)
969 opcode &= ~EWsGcOpFlagDrawOp;
973 LOG_PLAYBACK_GC_COMMAND(opcode, buf.Ptr())
974 CPlaybackGc::Instance()->CommandL(static_cast<TWsGcOpcodes>(opcode),buf);
979 pos += header.iCmdLength; // Move on, header indicates length
982 CleanupStack::PopAndDestroy(annoObs);
985 LOG_WINDOW_DRAW_COMMANDS_END(WsWin());
988 /*------------------------------------------------------------------------------
989 Description: Called when the currently stored graphics commands
990 are no longer required.
991 -----------------------------------------------------------------------------*/
992 void CWsRedrawMsgWindow::DiscardStoredCommands()
994 iCurrentSegment = NULL;
995 if (iRedrawSegments.Count() > 0)
997 // First of all, if we have any redraws pending, update the screen with
998 // whatever commands we have before we throw them away:
999 if (iFlags & EPendingScheduledDraw)
1000 Screen()->DoRedrawNow();
1002 // for all regions or just Partial Redraw regions > index 0: delete bitmaps and draw commands
1003 iRedrawSegments.ResetAndDestroy();
1009 void CWsRedrawMsgWindow::CreateNewSegmentL(const TRect& aRect, TRedrawSegmentType aNewRedrawRegionType)
1011 CWsRedrawMsgWindow::CRedrawSegment* newRegion = CWsRedrawMsgWindow::CRedrawSegment::NewLC(aRect, aNewRedrawRegionType, *this);
1013 iRedrawSegments.AppendL(newRegion);
1014 iCurrentSegment = newRegion;
1015 CleanupStack::Pop(newRegion);
1017 // Set iLastDrawGc to NULL. This will cause all GC attributes to be stored
1018 // in redraw store when the window receives the next command
1022 void CWsRedrawMsgWindow::AddFbsBitmapsL(TInt aHandle, TInt aMaskHandle)
1024 AddFbsBitmapRefL(aHandle);
1027 AddFbsBitmapRefL(aMaskHandle);
1031 void CWsRedrawMsgWindow::AddFbsBitmapRefL(TInt aHandle)
1033 TInt indexBitmapRef = iFbsBitmapRefArray.FindInOrder(aHandle, &FindBitmapRefByHandle);
1035 // check whether the window already has a reference for the bitmap through a segment other than the current one
1036 if (indexBitmapRef >=0)
1038 TBool handleAppend = iCurrentSegment->AppendNonDuplicateBitmapHandleL(aHandle);
1041 iFbsBitmapRefArray[indexBitmapRef]->IncRefCount();
1046 CFbsBitmapRef* bitmapRef = new(ELeave) CFbsBitmapRef;
1047 CleanupStack::PushL(bitmapRef);
1048 if (bitmapRef->Duplicate(aHandle)!=KErrNone)
1049 OwnerPanic(EWservPanicBitmap);
1050 iFbsBitmapRefArray.InsertInOrderL(bitmapRef, TLinearOrder<CFbsBitmapRef>(InsertBitmapRefByHandle));
1051 CleanupStack::Pop(bitmapRef);
1052 bitmapRef->IncRefCount();
1054 TBool bitmapHandleAppended = EFalse;
1055 TRAPD(err, bitmapHandleAppended = iCurrentSegment->AppendNonDuplicateBitmapHandleL(aHandle);
1056 WS_ASSERT_DEBUG(bitmapHandleAppended,EWsPanicUnexpectedBitmapHandleInArray););
1058 TRAPD(err, iCurrentSegment->AppendNonDuplicateBitmapHandleL(aHandle););
1060 if (err != KErrNone)
1062 // Cleanup the array, then propagate the error
1063 indexBitmapRef = iFbsBitmapRefArray.FindInOrder(aHandle, &FindBitmapRefByHandle);
1064 if (indexBitmapRef >= 0)
1066 iFbsBitmapRefArray[indexBitmapRef]->DecRefCount();
1067 delete iFbsBitmapRefArray[indexBitmapRef];
1068 iFbsBitmapRefArray.Remove(indexBitmapRef);
1075 // Only called during playback of a redraw store segment
1076 CFbsBitmap* CWsRedrawMsgWindow::BitmapFromHandle(TInt aHandle) const
1078 const TInt index = iFbsBitmapRefArray.FindInOrder(aHandle, &FindBitmapRefByHandle);
1079 if (index != KErrNotFound)
1081 CFbsBitmap* bitmap = iFbsBitmapRefArray[index];
1088 void CWsRedrawMsgWindow::AddWsBitmapsL(TInt aHandle, TInt aMaskHandle)
1090 if (iWsWin->WsOwner() == NULL)
1091 Panic(EWsPanicDrawCommandsInvalidState);
1092 DWsBitmap * bmp = static_cast<DWsBitmap*>(iWsWin->WsOwner()->HandleToObj(aHandle, WS_HANDLE_BITMAP));
1094 OwnerPanic(EWservPanicBitmap);
1095 iCurrentSegment->AddWsBitmapL(bmp);
1098 bmp = static_cast<DWsBitmap*>(iWsWin->WsOwner()->HandleToObj(aMaskHandle, WS_HANDLE_BITMAP));
1100 OwnerPanic(EWservPanicBitmap);
1101 iCurrentSegment->AddWsBitmapL(bmp);
1105 void CWsRedrawMsgWindow::CRedrawSegment::AddWsBitmapL(DWsBitmap* bitmap)
1107 iWsBitmapArray.AppendL(bitmap);
1108 bitmap->IncRefCount();
1111 void CWsRedrawMsgWindow::AddWsFontL(TInt aHandle)
1113 if (iWsWin->WsOwner()==NULL)
1114 Panic(EWsPanicDrawCommandsInvalidState);
1115 TDblQueIter<CWsFbsFont> iter(CWsFontCache::List());
1116 CWsFbsFont* font=NULL;
1117 while((font=iter++)!=NULL)
1119 if (font->Handle()==aHandle)
1124 iCurrentSegment->iWsFontArray.AppendL(font);
1129 void CWsRedrawMsgWindow::CRedrawSegment::AddDrawerL(TGraphicDrawerId aDrawerId)
1131 TInt error = iDrawerArray.InsertInOrder(aDrawerId, TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare));
1132 if (error != KErrAlreadyExists && error != KErrNone)
1138 TBool CWsRedrawMsgWindow::CRedrawSegment::ContainsDrawers(const TArray<TGraphicDrawerId>& aDrawers,const TRegion& aRegion) const
1140 TBool result = EFalse;
1141 if (iDrawerArray.Count() > 0)
1143 STACK_REGION tempRegion;
1144 tempRegion.Intersection(iRegion, aRegion);
1145 if (tempRegion.CheckError() || (tempRegion.Count() > 0) )
1146 { // regions do intersect, (presumed if region had an error); so check for a matching Id
1147 const TInt drawersCount = aDrawers.Count();
1148 for (TInt idx = 0; idx < drawersCount; ++idx)
1149 { // (iDrawerArray is kept sorted)
1150 if (KErrNotFound != iDrawerArray.FindInOrder(aDrawers[idx], TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare)))
1156 const TInt count = iDrawerArray.Count();
1157 for(TInt i = 0; i < count; i++)
1159 const CWsGraphicDrawer* drawer = CWsTop::WindowServer()->ResolveGraphic(iDrawerArray[i]);
1160 if(drawer && drawer->Contains(aDrawers))
1173 void CWsRedrawMsgWindow::AddWsDrawableSourceL(TInt aHandle)
1175 if (iWsWin->WsOwner() == NULL)
1176 Panic(EWsPanicDrawCommandsInvalidState);
1177 CWsDrawableSource* drawableSource = static_cast<CWsDrawableSource*>(iWsWin->WsOwner()->HandleToObj(aHandle, WS_HANDLE_DRAWABLE_SOURCE));
1178 if (!drawableSource)
1179 OwnerPanic(EWservPanicDrawableSource);
1180 iCurrentSegment->AddWsDrawableSourceL(drawableSource);
1183 void CWsRedrawMsgWindow::CRedrawSegment::AddWsDrawableSourceL(CWsDrawableSource* aDrawableSource)
1185 iWsDrawableSourceArray.AppendL(aDrawableSource);
1186 aDrawableSource->IncRefCount();
1189 inline TBool CWsRedrawMsgWindow::NoBuffer() const
1191 return (iRedrawSegments.Count() == 0);
1194 void CWsRedrawMsgWindow::ClientExposing()
1199 /*------------------------------------------------------------------------------
1200 Description: If a complete set of drawing commands have been stored
1201 this method attempts to draw ALL the commands via DrawCommandsL().
1202 It also draws the window in the background colour if the window is
1204 -----------------------------------------------------------------------------*/
1205 void CWsRedrawMsgWindow::DrawWindow()
1207 iFlags &= ~EPendingScheduledDraw;
1208 // This is a happy window - it can draw itself whenever we ask.
1209 CScreen* screen = Screen();
1210 if ((screen->AutoClear() && (iFlags & EBackgroundClear)) || HasElement())
1212 DrawBackgroundColor(*iRedrawRegion,(iFlags&EBackgroundClear)!=0);
1216 screen->WindowElements().UnassignPlacedElements(*iRedrawRegion,*CliWin(),CPlaybackGc::Instance()->GcDrawingCount());
1218 // If valid commands have been stored, draw them.
1219 if (iRedrawSegments.Count() > 0)
1222 TRAP_IGNORE(DrawCommandsL());
1225 if (HasElement()) //HasElement won't be cleared following first call above. May get set...
1227 TInt state=screen->WindowElements().CleanUpPlacedElements(*CliWin(),CPlaybackGc::Instance()->GcDrawingCount());
1228 if (state&CWindowElement::EFastPath)
1232 screen->ElementAdded();
1236 screen->ElementRemoved();
1242 void CWsRedrawMsgWindow::RemoveFromRedrawQueueIfEmpty()
1244 if (iInvalid.Count()==0)
1246 iInvalid.Clear(); // Ensures heap cell is freed, otherwise may be left as an empty cell
1247 iWsWin->WsOwner()->RedrawQueue()->RemoveInvalid(this);
1251 const TRegion& CWsRedrawMsgWindow::InvalidArea() const
1256 TBool CWsRedrawMsgWindow::NeedsRedraw() const
1257 // If iInvalid has an persistant error it will not be reported as needing a redraw,
1258 // this is needed as otherwise cases where validation of a window results
1259 // in iInvalid having an error will get into an endless cycle of redraws.
1260 // The down side of this is that sometimes a window will not be sent a redraw
1261 // message when it needs it, some things can't be perfect!
1264 if ((!iWsWin->IsVisible()) || iInvalid.IsEmpty())
1267 TRect nextRedrawRect;
1268 return GetRedrawRect(nextRedrawRect);
1271 TBool CWsRedrawMsgWindow::GetRedrawRect(TRect &aRect) const
1273 if (iWsWin->ClientSetInvisible())
1278 aRect = iRedrawRect;
1279 return (!aRect.IsEmpty());
1281 else if(iInvalid.CheckError())
1283 if (Screen()->ChangeTracking())
1285 aRect = iWsWin->AbsRect();
1289 if (iFlags & EStoringEntireWindow || iWsWin->VisibleRegion().CheckError())
1291 aRect = iWsWin->AbsRect();
1295 aRect = iWsWin->VisibleRegion().BoundingRect();
1298 if (!(iFlags & EStoringEntireWindow))
1299 iWsWin->ClipRectToViewport(aRect);
1300 aRect.Move(-iWsWin->Origin());
1301 return (!aRect.IsEmpty());
1303 else if(iInvalid.Count())
1305 if (iFlags & EStoringEntireWindow)
1307 aRect = iInvalid.BoundingRect();
1312 region.Copy(iInvalid);
1313 region.Offset(iWsWin->Origin());
1314 if (!Screen()->ChangeTracking())
1316 region.Intersect(iWsWin->VisibleRegion());
1318 if (region.CheckError())
1320 aRect = iInvalid.BoundingRect();
1321 aRect.Move(iWsWin->Origin());
1325 aRect = region.BoundingRect();
1328 iWsWin->ClipRectToViewport(aRect);
1329 aRect.Move(-iWsWin->Origin());
1331 return (!aRect.IsEmpty());
1339 const TRegion &CWsRedrawMsgWindow::BaseDrawRegion() const
1341 return(iWsWin->VisibleRegion());
1344 void CWsRedrawMsgWindow::ClipInvalidRegion(const TRect &aRect)
1346 if (iInvalid.Count()>0)
1348 iInvalid.ClipRect(aRect);
1349 RemoveFromRedrawQueueIfEmpty();
1353 void CWsRedrawMsgWindow::EndRedraw()
1356 OwnerPanic(EWservPanicDrawCommandsInvalidState);
1357 if (iCurrentSegment)
1359 iCurrentSegment->iDrawCommands->Compress();
1361 PromotePendingSegment();
1363 // Schedule an update of the area of the screen we just drew to:
1364 iFlags |= EPendingScheduledDraw;
1365 if (Screen()->ChangeTracking())
1367 iWsWin->AddDirtyWindowRegion(iCurrentSegment->iRegion); // stored in window coordinates
1368 if (iWsWin->IsVisible())
1370 // Window is visible, (we're ignoring whether it's obscured or not)
1371 // we need to send the new draw commands to the render stage.
1372 Screen()->ScheduleWindow(iWsWin);
1375 if (!iWsWin->HasBeenDrawnToScreen())
1377 CliWin()->ScheduleRegionUpdate(NULL);
1379 else if(!Screen()->ChangeTracking() && (iWsWin->VisibleRegion().Count() || iWsWin->VisibleRegion().CheckError())) // on screen? good.
1381 STACK_REGION draw; //### in low memory where VisibleRegion() is intact we can degrade much better than this!
1382 draw.Copy(iCurrentSegment->iRegion);
1383 draw.Offset(iWsWin->Origin());
1384 draw.Intersect(iWsWin->VisibleRegion());
1385 if(!draw.CheckError())
1386 Screen()->AddRedrawRegion(draw);
1388 Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
1393 iCurrentSegment = NULL;
1395 //store the result of the current redraw
1396 if(iFlags&ECurrentRedrawFailedStorage)
1397 iFlags |= EPreviousRedrawFailedStorage; //set
1399 iFlags &= ~EPreviousRedrawFailedStorage; //unset
1400 iFlags &= ~ECurrentRedrawFailedStorage; //unset the flag for the next redraw
1403 iFlags&=~EBeginEndRedraw;
1406 void CWsRedrawMsgWindow::ValidateRect(const TRect *aRect)
1408 if (!WsWin()->BaseParent())
1409 OwnerPanic(EWservPanicParentDeleted);
1411 iRedrawRect = *aRect;
1412 if (!iInvalid.IsEmpty())
1414 STACK_REGION validated;
1415 validated.Copy(iInvalid);
1417 validated.ClipRect(iRedrawRect);
1419 if (iInvalid.CheckError())
1421 iInvalid.Copy(iWsWin->VisibleRegion());
1422 iInvalid.Offset(-iWsWin->Origin());
1424 iInvalid.SubRegion(validated);
1427 RemoveFromRedrawQueueIfEmpty();
1430 TRgb CWsRedrawMsgWindow::BackColor() const
1436 This function used to be quite clever about what it invalidated and what it redrew by copying
1437 rectangles of the screen around. This is a lot less subtle, and makes calling Scroll pretty much
1438 pointless, but it IS functionally correct.
1440 void CWsRedrawMsgWindow::Scroll(const TRect &aClipRect, const TPoint &aOffset,const TRect &aRect)
1443 rect.Intersection(aClipRect);
1447 rect.Intersection(aClipRect);
1451 void CWsRedrawMsgWindow::ClearRedrawStore(TBool aClearPendingRedraw)
1453 if(aClearPendingRedraw && (iFlags & EPendingScheduledDraw))
1454 iFlags &= ~EPendingScheduledDraw;
1456 DiscardStoredCommands();
1461 void CWsRedrawMsgWindow::PrepareForResizeL(const TSize& aSize, TSize& /*aOldSize*/)
1463 TBool anyIncreases(EFalse);
1464 if (aSize.iWidth>iWsWin->Size().iWidth||aSize.iHeight>iWsWin->Size().iHeight)
1466 anyIncreases = ETrue;
1469 TRect newWinRect(TPoint(0,0),aSize);
1470 iInvalid.ClipRect(newWinRect);
1473 // add new invalid region to iInvalid
1474 iInvalid.AddRect(newWinRect);
1476 iWsWin->WsOwner()->TriggerRedraw();
1480 void CWsRedrawMsgWindow::Moved()
1482 if (!(iFlags & EStoringEntireWindow))
1484 DiscardSegmentsOutsideViewport();
1486 if (iInvalid.Count())
1489 iWsWin->WsOwner()->TriggerRedraw();
1493 TBool CWsRedrawMsgWindow::Contains(const TArray<TGraphicDrawerId>& aDrawers,const TRegion& aRegion) const
1495 if (iRedrawSegments.Count() > 0)
1497 // scan redraw store: calls Contains() on every region drawing commands are stored for,
1498 // looking for a DrawWsGraphic command that intersects the aRegion
1499 TBool contains = EFalse;
1500 const TInt regionCount = iRedrawSegments.Count();
1502 // Apply an origin correction. The input aRegion is screen-absolute, while the redraw regions are window-relative.
1503 STACK_REGION relRegion;
1504 relRegion.Copy(aRegion);
1505 relRegion.Offset(-(CliWin()->Origin().iX), -(CliWin()->Origin().iY));
1507 // loop through regions, stops when a match is found
1508 for (TInt regionNum = 0; (regionNum < regionCount) && !contains; ++regionNum)
1510 contains = iRedrawSegments[regionNum]->ContainsDrawers(aDrawers, relRegion);
1517 return CWsWindowRedraw::Contains(aDrawers,aRegion);
1521 TBool CWsRedrawMsgWindow::RedrawingInProgress() const
1523 return (iFlags & EBeginEndRedraw);
1526 void CWsRedrawMsgWindow::WindowClosing()
1528 iWsWin->WsOwner()->RedrawQueue()->RemoveInvalid(this);
1529 CWsWindowRedraw::WindowClosing();
1532 TBool CWsRedrawMsgWindow::IsRedrawStoreEmpty() const
1534 return (iRedrawSegments.Count() <= 0); // Begin and End redraw are in the store too.
1537 TBool CWsRedrawMsgWindow::IsBackgroundClearEnabled() const
1539 return ((iFlags & EBackgroundClear) != 0);
1542 void CWsRedrawMsgWindow::CleanupBitmapRefArray(const RArray<TInt>& aHandleArray)
1544 TInt count = aHandleArray.Count();
1545 for(TInt ii = count - 1 ; ii >= 0; ii--)
1547 const TInt indexBitmapRef = iFbsBitmapRefArray.FindInOrder(aHandleArray[ii], &FindBitmapRefByHandle);
1548 WS_ASSERT_DEBUG(indexBitmapRef >= 0,EWsPanicUnexpectedBitmapHandleInArray);
1549 if (indexBitmapRef>=0)
1551 iFbsBitmapRefArray[indexBitmapRef]->DecRefCount();
1552 if (iFbsBitmapRefArray[indexBitmapRef]->RefCount()==0)
1554 delete iFbsBitmapRefArray[indexBitmapRef];
1555 iFbsBitmapRefArray.Remove(indexBitmapRef);
1561 void CWsRedrawMsgWindow::SetScope(TScope aScope)
1563 if (aScope == EStoreEntireWindow)
1565 if (!(iFlags & EStoringEntireWindow))
1567 iFlags |= EStoringEntireWindow;
1573 if (iFlags & EStoringEntireWindow)
1575 iFlags &= ~ EStoringEntireWindow;
1576 DiscardSegmentsOutsideViewport();
1582 Removes all segments from the redraw store which are outside the viewport onto the window.
1583 Note that this doesn't clip the regions of those segments which are partly outside, since
1584 this wouldn't actually achieve anything useful.
1586 This function allocates memory so it is not suitable to run as part of ReleaseMemory.
1588 TBool CWsRedrawMsgWindow::DiscardSegmentsOutsideViewport()
1590 TBool discarded = EFalse;
1591 TInt count = iRedrawSegments.Count();
1592 STACK_REGION viewport;
1593 CliWin()->GetClippedBaseArea(viewport);
1594 viewport.Offset(-iWsWin->Origin());
1595 STACK_REGION intersect;
1596 for (TInt idx = count - 1; idx >= 0; --idx)
1598 CRedrawSegment * segment = iRedrawSegments[idx];
1599 intersect.Intersection(segment->iRegion, viewport);
1600 if (!intersect.CheckError() && intersect.IsEmpty())
1602 iInvalid.Union(segment->iRegion);
1604 iRedrawSegments.Remove(idx);
1605 if (iCurrentSegment == segment)
1606 iCurrentSegment = NULL;
1616 Statements encapsulated in between Lock() and Unlock() is guaranteed to execute in an
1617 atomic way without being interupted by a call to ReleaseMemory from CWsMemoryManager.
1618 Locking will prevent memory belonging to this object to be freed during a
1619 memory alloc/realloc originating from self.
1621 void CWsRedrawMsgWindow::Lock()
1626 void CWsRedrawMsgWindow::Unlock()
1629 WS_ASSERT_DEBUG(iMemoryLock >= 0, EWsPanicMemoryLock);
1632 TBool CWsRedrawMsgWindow::ReleaseMemory(MWsMemoryRelease::TMemoryReleaseLevel aLevel)
1634 //When this function is called, wserv is in the middle of executing something.
1635 //Therefore we can not safely do anything that alters the state of any shared
1636 //resouces (like e.g. CScreenRedraw::iInvalid).
1637 //In addition, we should refrain from anything that might try to allocate memory.
1638 TBool released = EFalse;
1639 //Don't release iRedrawSegments from this win if its currently being rendered,
1640 //is releasing memory or is receiving drawcommands.
1641 if (iMemoryLock == 0 && !iCurrentSegment)
1646 case MWsMemoryRelease::ELow:
1648 case MWsMemoryRelease::EMedium:
1650 case MWsMemoryRelease::EHigh:
1651 //Only release memory from background windows.
1652 if (iRedrawSegments.Count() > 0 && iWsWin->VisibleRegion().IsEmpty())
1654 ReleaseRedrawSegments();
1664 void CWsRedrawMsgWindow::ReleaseRedrawSegments()
1667 iCurrentSegment = NULL;
1668 iRedrawSegments.ResetAndDestroy();
1670 //The call to ResetAndDestroy just freed some memory so it should be
1671 //possible to call Invalidate() now.
1674 //Releasing the same window over and over again could quickly end up in
1675 //a never ending loop with a high-prio client before we find the window
1676 //that has nicked all memory. So call accessed now to prevent that.
1680 void CWsRedrawMsgWindow::VisibleRegionChange()
1682 if (!iFlags & EStoringEntireWindow)
1684 DiscardSegmentsOutsideViewport();
1686 if ((!iInvalid.IsEmpty()) && (!iWsWin->VisibleRegion().IsEmpty()))
1688 STACK_REGION exposed;
1689 exposed.Copy(iInvalid);
1690 exposed.Offset(iWsWin->Origin());
1691 exposed.Intersect(iWsWin->VisibleRegion());
1692 if (!exposed.IsEmpty())
1700 TBool CWsRedrawMsgWindow::ReadyToDraw() const
1702 //We are only ready to draw when we have a complete segment.
1703 if (iWsWin->HasBeenDrawnToScreen())
1706 if (iRedrawSegments.Count() == 0)
1709 if (iRedrawSegments.Count() > 1)
1712 if (iRedrawSegments[0]->iRedrawSegmentType == ESegmentTypePendingRedraw)
1718 TInt CWsRedrawMsgWindow::SizeInBytes() const
1720 TInt size = sizeof(CWsRedrawMsgWindow);
1721 for(TInt i = iRedrawSegments.Count()-1; i >= 0; i--)
1723 size += iRedrawSegments[i]->SizeInBytes();
1725 size += iInvalid.Count() * sizeof(TRect);
1726 size += iLocalRedrawRegion.Count() * sizeof(TRect);
1730 void CWsRedrawMsgWindow::PromotePendingSegment()
1732 if (iRedrawSegments.Count() > 0 && iRedrawSegments[iRedrawSegments.Count() - 1]->iRedrawSegmentType == ESegmentTypePendingRedraw)
1734 CRedrawSegment * segment = iRedrawSegments[iRedrawSegments.Count() - 1];
1735 const TRect * rect = segment->iRegion.RectangleList();
1736 // when we get here there should only ever be one rectangle in the region, but we are playing safe
1737 for (TInt r = 0; r < segment->iRegion.Count(); ++r)
1739 SubtractRectFromSegmentArray(*rect);
1742 segment->iRedrawSegmentType = ESegmentTypeRedraw;
1746 CFbsBitmapRef::CFbsBitmapRef()
1750 CFbsBitmapRef::~CFbsBitmapRef()
1752 WS_ASSERT_DEBUG(iRefCount==0,EWsPanicCounterValue);