1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nga/SERVER/openwfc/redrawmsgwindow.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,1753 @@
1.4 +// Copyright (c) 1995-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +// Window redraw code, three sorts of redrawing are supported
1.18 +// CRedrawMsgWindow handles it via sending a redraw message to the client
1.19 +//
1.20 +//
1.21 +
1.22 +#include "redrawmsgwindow.h"
1.23 +#include "gc.h"
1.24 +#include "playbackgc.h"
1.25 +#include "inifile.h"
1.26 +#include "rootwin.h"
1.27 +#include "wstop.h"
1.28 +#include "ANIM.H"
1.29 +#include "EVQUEUE.H"
1.30 +#include <s32mem.h>
1.31 +#include <gdi.h>
1.32 +#include "panics.h"
1.33 +#include "rootwin.h"
1.34 +#include "EVENT.H"
1.35 +#include "wsfont.h"
1.36 +#include <graphics/WSGRAPHICDRAWERINTERFACE.H>
1.37 +#include <graphics/surface.h>
1.38 +#include "windowelementset.h"
1.39 +
1.40 +#include "drawresource.h"
1.41 +#include "../debuglog/DEBUGLOG.H"
1.42 +
1.43 +const TUint KDrawBufferGranularity = 240;
1.44 +const TInt KRedrawRegionGranularity = 8;
1.45 +const TInt KReadBufferMaxLen=0x100;
1.46 +const TInt KRegionCompressThreshold = 6; // number of rectangles in region at which we try to compress it
1.47 +
1.48 +TBool CWsRedrawMsgWindow::iAtomic = EFalse;
1.49 +
1.50 +extern CDebugLogBase *wsDebugLog;
1.51 +
1.52 +#ifndef _DEBUG
1.53 +
1.54 +#define LOG_WINDOW_DRAW_COMMANDS_START(wswin,region)
1.55 +#define LOG_WINDOW_DRAW_COMMANDS_END(wswin)
1.56 +#define LOG_REDRAW_SEGMENT(segmentIndex,segmentType)
1.57 +#define LOG_REDRAW_SEGMENT_REGION(region)
1.58 +#define LOG_PLAYBACK_GC_COMMAND(opcode,data)
1.59 +
1.60 +#else
1.61 +
1.62 +#define LOG_WINDOW_DRAW_COMMANDS_START(wswin,region) LogDrawCommandsStart(wswin,region)
1.63 +#define LOG_WINDOW_DRAW_COMMANDS_END(wswin) LogDrawCommandsEnd(wswin)
1.64 +#define LOG_REDRAW_SEGMENT(segmentIndex,segmentType) LogRedrawSegment(segmentIndex, segmentType)
1.65 +#define LOG_REDRAW_SEGMENT_REGION(region) {if(wsDebugLog){ LogRegion(region);}}
1.66 +#define LOG_PLAYBACK_GC_COMMAND(opcode,data) {if (wsDebugLog) {wsDebugLog->Command(WS_HANDLE_GC, opcode, data, NULL);}}
1.67 +
1.68 +LOCAL_C void LogRedrawSegment(TUint aSegmentIndex, CWsRedrawMsgWindow::TRedrawSegmentType aSegmentType)
1.69 + {
1.70 + if (wsDebugLog)
1.71 + {
1.72 + TBuf<LogTBufSize> log;
1.73 + TTruncateOverflow overflow;
1.74 + _LIT(KLogRedrawSegment, ">> CRedrawSegment[%d] ");
1.75 + log.AppendFormat(KLogRedrawSegment, &overflow, aSegmentIndex);
1.76 + _LIT(KLogRedrawSegmentPending, "Pending");
1.77 + _LIT(KLogRedrawSegmentRedraw, "Redraw");
1.78 + switch(aSegmentType)
1.79 + {
1.80 + case CWsRedrawMsgWindow::ESegmentTypePendingRedraw :
1.81 + log.AppendFormat(KLogRedrawSegmentPending, &overflow);
1.82 + break;
1.83 + case CWsRedrawMsgWindow::ESegmentTypeRedraw :
1.84 + log.AppendFormat(KLogRedrawSegmentRedraw, &overflow);
1.85 + break;
1.86 + default :
1.87 + {
1.88 + }
1.89 + }
1.90 + wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.91 + }
1.92 + }
1.93 +
1.94 +LOCAL_C void LogDrawCommandsStart(const CWsWindow* aWsWin, const TRegion* aRegion)
1.95 + {
1.96 + if (wsDebugLog)
1.97 + {
1.98 + _LIT(KLogDrawCommandsStart, ">> CWsRedrawMsgWindow::DrawCommandsL() [%S][app %d] RWindow[%d]");
1.99 + const TDesC& clientName = aWsWin->WsOwner()->Client().FullName();
1.100 + TBuf<LogTBufSize> log;
1.101 + TTruncateOverflow overflow;
1.102 + log.AppendFormat(KLogDrawCommandsStart, &overflow, &clientName, aWsWin->WsOwner()->ConnectionHandle(), aWsWin->LogHandle());
1.103 + wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.104 + LogRegion(aRegion);
1.105 + }
1.106 + }
1.107 +
1.108 +LOCAL_C void LogDrawCommandsEnd(const CWsWindow* aWsWin)
1.109 + {
1.110 + if (wsDebugLog)
1.111 + {
1.112 + _LIT(KLogDrawCommandsEnd, "<< CWsRedrawMsgWindow::DrawCommandsL() [%S][app %d] RWindow[%d]");
1.113 + const TDesC& clientName = aWsWin->WsOwner()->Client().FullName();
1.114 + TBuf<LogTBufSize> log;
1.115 + TTruncateOverflow overflow;
1.116 + log.AppendFormat(KLogDrawCommandsEnd, &overflow, &clientName, aWsWin->WsOwner()->ConnectionHandle(), aWsWin->LogHandle());
1.117 + wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, log);
1.118 + }
1.119 + }
1.120 +
1.121 +#endif
1.122 +
1.123 +//
1.124 +// Redraw windows //
1.125 +//
1.126 +
1.127 +CWsRedrawMsgWindow::CRedrawSegment::CRedrawSegment(CWsRedrawMsgWindow& aRedraw) : iRedraw(aRedraw)
1.128 + {
1.129 + }
1.130 +
1.131 +CWsRedrawMsgWindow::CRedrawSegment* CWsRedrawMsgWindow::CRedrawSegment::NewLC(const TRect& aRect, TRedrawSegmentType aNewRegionType, CWsRedrawMsgWindow& aRedraw)
1.132 + {
1.133 + CRedrawSegment* self = new (ELeave) CRedrawSegment(aRedraw);
1.134 + CleanupStack::PushL(self);
1.135 + self->ConstructL(aRect, aNewRegionType);
1.136 + return self;
1.137 + }
1.138 +
1.139 +void CWsRedrawMsgWindow::StaticInitL()
1.140 + {
1.141 + // Note that the wsini.ini setting NONREDRAWAGELIMIT is obselete in
1.142 + // Wserv2 NGA. The effective behaviour of the system is as if the
1.143 + // value was set to zero.
1.144 +
1.145 + _LIT(KAtomicRedraws, "ATOMICREDRAWS");
1.146 + if (WsIniFile->FindVar(KAtomicRedraws) ||
1.147 + WsIniFile->FindVar(KWSERVIniFileVarChangeTracking)) //ChangeTracking requires atomic redraws to be enabled to
1.148 + iAtomic = ETrue; //prevent playing redraw-segemts before they are completed.
1.149 + else
1.150 + iAtomic = EFalse;
1.151 + }
1.152 +
1.153 +static TInt FindBitmapRefByHandle(const TInt* aKey, const CFbsBitmapRef& aBitmapRef)
1.154 + { // compare handles
1.155 + return *aKey - aBitmapRef.Handle();
1.156 + }
1.157 +
1.158 +static TInt InsertBitmapRefByHandle(const CFbsBitmapRef& aFirst, const CFbsBitmapRef& aSecond)
1.159 + {
1.160 + return aFirst.Handle() - aSecond.Handle();
1.161 + }
1.162 +
1.163 +void CWsRedrawMsgWindow::CRedrawSegment::ReleaseFontsBitmapsDrawableSources()
1.164 + {
1.165 + // release Bitmap, Font and Drawable Source handles
1.166 + TInt count = iWsBitmapArray.Count();
1.167 + TInt ii;
1.168 + for (ii = count - 1; ii >= 0; ii--)
1.169 + {
1.170 + iWsBitmapArray[ii]->DecRefCount();
1.171 + iWsBitmapArray.Remove(ii);
1.172 + }
1.173 +
1.174 + count = iWsFontArray.Count();
1.175 + for (ii = count - 1; ii >= 0; --ii)
1.176 + {
1.177 + CWsFontCache::Instance()->ReleaseFont(iWsFontArray[ii]);
1.178 + iWsFontArray.Remove(ii);
1.179 + }
1.180 +
1.181 + iRedraw.CleanupBitmapRefArray(iBitmapHandleArray);
1.182 + iBitmapHandleArray.Close();
1.183 +
1.184 + count = iWsDrawableSourceArray.Count();
1.185 + for(ii = count - 1 ; ii >= 0; ii--)
1.186 + {
1.187 + iWsDrawableSourceArray[ii]->DecRefCount();
1.188 + iWsDrawableSourceArray.Remove(ii);
1.189 + }
1.190 + }
1.191 +
1.192 +/* Set new rectangle and region type for initial or reset redraw region
1.193 + @leave KErrNoMemory no memory to update region details
1.194 + */
1.195 +void CWsRedrawMsgWindow::CRedrawSegment::ConstructL(const TRect& aRect, TRedrawSegmentType aNewRegionType)
1.196 + {
1.197 + iDrawCommands = CBufSeg::NewL(KDrawBufferGranularity);
1.198 + iCreationTime.UniversalTime();
1.199 +
1.200 + iRegion.AddRect(aRect);
1.201 + if (iRegion.CheckError())
1.202 + {
1.203 + User::Leave(KErrNoMemory);
1.204 + }
1.205 + iRedrawSegmentType = aNewRegionType;
1.206 + }
1.207 +
1.208 +CWsRedrawMsgWindow::CRedrawSegment::~CRedrawSegment()
1.209 + {
1.210 +
1.211 + if (iDrawCommands)
1.212 + {
1.213 + delete iDrawCommands;
1.214 + }
1.215 + iRegion.Close();
1.216 + // Release Font, Bitmap and Drawable Source handles, close arrays
1.217 + ReleaseFontsBitmapsDrawableSources();
1.218 + iWsBitmapArray.Close();
1.219 + iWsFontArray.Close();
1.220 + iDrawerArray.Close();
1.221 + iWsDrawableSourceArray.Close();
1.222 + }
1.223 +
1.224 +
1.225 +TInt CWsRedrawMsgWindow::CRedrawSegment::SizeInBytes() const
1.226 + {
1.227 + TInt size = sizeof(CWsRedrawMsgWindow::CRedrawSegment);
1.228 + size += iDrawCommands->Size();
1.229 + size += iBitmapHandleArray.Count() * sizeof(TInt);
1.230 + size += iWsBitmapArray.Count() * sizeof(DWsBitmap);
1.231 + size += iWsFontArray.Count() * sizeof(CWsFbsFont);
1.232 + size += iDrawerArray.Count() * sizeof(TGraphicDrawerId);
1.233 + size += iWsDrawableSourceArray.Count() * sizeof(CWsDrawableSource);
1.234 + return size;
1.235 + }
1.236 +
1.237 +TBool CWsRedrawMsgWindow::CRedrawSegment::AppendNonDuplicateBitmapHandleL(TInt aHandle)
1.238 + {
1.239 + TBool handleAppend = EFalse;
1.240 + TInt indexBitmapHandle = iBitmapHandleArray.Find(aHandle);
1.241 + if (indexBitmapHandle<0)
1.242 + {
1.243 + iBitmapHandleArray.AppendL(aHandle);
1.244 + handleAppend = ETrue;
1.245 + }
1.246 + return handleAppend;
1.247 + }
1.248 +
1.249 +CWsRedrawMsgWindow::CWsRedrawMsgWindow(CWsWindow *aWin)
1.250 + :CWsWindowRedraw(aWin), iBackColor(aWin->RootWindow()->DefaultBackgroundColor()), iFlags(EBackgroundClear|EStoringEntireWindow),
1.251 + iRedrawSegments(KRedrawRegionGranularity),
1.252 + iCurrentSegment(0),
1.253 + iOSBStatus(ETrue)
1.254 + {
1.255 + }
1.256 +
1.257 +void CWsRedrawMsgWindow::ConstructL()
1.258 + {
1.259 + CWsWindowRedraw::ConstructL();
1.260 + Invalidate(&WsWin()->Rel());
1.261 + iWsWin->WsOwner()->RedrawQueue()->ReCalcOrder();
1.262 + }
1.263 +
1.264 +CWsRedrawMsgWindow::~CWsRedrawMsgWindow()
1.265 + {
1.266 + WS_ASSERT_DEBUG(iMemoryLock == 0, EWsPanicMemoryLock);
1.267 + RemoveFromRedrawQueueIfEmpty();
1.268 + iInvalid.Close();
1.269 + iLocalRedrawRegion.Close();
1.270 + iRedrawSegments.ResetAndDestroy();
1.271 + TInt count = iFbsBitmapRefArray.Count();
1.272 + WS_ASSERT_DEBUG(count==0,EWsPanicBitmapArrayNotEmpty);
1.273 + iFbsBitmapRefArray.ResetAndDestroy();
1.274 + }
1.275 +
1.276 +/**
1.277 +These three functions actually check for a value they have already asserted on. This is intentional.
1.278 +*/
1.279 +void CWsRedrawMsgWindow::ExpandCommandBufferL(TInt aLength)
1.280 + {
1.281 + WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
1.282 +
1.283 + if (iCurrentSegment)
1.284 + {
1.285 + // need more space?
1.286 + if (iCurrentSegment->iCurrentCommandBufferWritePos + aLength > iCurrentSegment->iDrawCommands->Size())
1.287 + {
1.288 + iCurrentSegment->iDrawCommands->ResizeL(iCurrentSegment->iCurrentCommandBufferWritePos + aLength);
1.289 + }
1.290 + }
1.291 + }
1.292 +
1.293 +void CWsRedrawMsgWindow::CommandBufferWrite(const TDesC8& aDes, TInt aLength)
1.294 + {
1.295 + WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
1.296 + WS_ASSERT_DEBUG(iCurrentSegment->iCurrentCommandBufferWritePos + aLength <= iCurrentSegment->iDrawCommands->Size(), EWsPanicDrawCommandsInvalidState);
1.297 + if (iCurrentSegment)
1.298 + {
1.299 + iCurrentSegment->iDrawCommands->Write(iCurrentSegment->iCurrentCommandBufferWritePos, aDes, aLength);
1.300 + iCurrentSegment->iCurrentCommandBufferWritePos += aLength;
1.301 + }
1.302 + }
1.303 +
1.304 +void CWsRedrawMsgWindow::CommandBufferWrite(const TAny* aPtr,TInt aLength)
1.305 + {
1.306 + WS_ASSERT_DEBUG(iCurrentSegment != NULL, EWsPanicRedrawSegmentsInvalidState);
1.307 + WS_ASSERT_DEBUG(iCurrentSegment->iCurrentCommandBufferWritePos + aLength <= iCurrentSegment->iDrawCommands->Size(), EWsPanicDrawCommandsInvalidState);
1.308 + if (iCurrentSegment)
1.309 + {
1.310 + iCurrentSegment->iDrawCommands->Write(iCurrentSegment->iCurrentCommandBufferWritePos, aPtr, aLength);
1.311 + iCurrentSegment->iCurrentCommandBufferWritePos += aLength;
1.312 + }
1.313 + }
1.314 +
1.315 +/*------------------------------------------------------------------------------
1.316 + Description: Processes draw commands. These are received as opcodes.
1.317 + -----------------------------------------------------------------------------*/
1.318 +TBool CWsRedrawMsgWindow::CommandL(TInt aOpcode, TWsWinCmdUnion &aCmd)
1.319 + {
1.320 + switch(aOpcode)
1.321 + {
1.322 + case EWsWinOpEnableOSB:
1.323 + iOSBStatus = ETrue;
1.324 + break;
1.325 + case EWsWinOpDisableOSB:
1.326 + iOSBStatus = EFalse;
1.327 + break;
1.328 + case EWsWinOpSetBackgroundColor:
1.329 + iBackColor = *aCmd.rgb;
1.330 + iFlags |= EBackgroundClear;
1.331 + break;
1.332 + case EWsWinOpSetNoBackgroundColor:
1.333 + iFlags &= ~EBackgroundClear;
1.334 + break;
1.335 + case EWsWinOpInvalidate:
1.336 + Invalidate(aCmd.rect);
1.337 + break;
1.338 + case EWsWinOpInvalidateFull:
1.339 + Invalidate();
1.340 + break;
1.341 + case EWsWinOpBeginRedraw:
1.342 + BeginRedraw(aCmd.rect);
1.343 + ValidateRect(aCmd.rect);
1.344 + break;
1.345 + case EWsWinOpBeginRedrawFull:
1.346 + BeginRedraw(NULL);
1.347 + ValidateRect(NULL);
1.348 + break;
1.349 + case EWsWinOpEndRedraw:
1.350 + EndRedraw();
1.351 + break;
1.352 + case EWsWinOpGetInvalidRegionCount:
1.353 + {
1.354 + SetReply(iInvalid.Count());
1.355 + }
1.356 + break;
1.357 + case EWsWinOpGetInvalidRegion:
1.358 + {
1.359 + if ((*aCmd.Int) <= 0)
1.360 + OwnerPanic(EWservPanicInvalidRegionCount);
1.361 + if ((!iInvalid.CheckError()) && iInvalid.Count() == (*aCmd.Int))
1.362 + {
1.363 + CWsClient::ReplyBuf(iInvalid.RectangleList(),(*aCmd.Int) * sizeof(TRect));
1.364 + SetReply(EFalse);
1.365 + }
1.366 + else
1.367 + SetReply(ETrue);
1.368 + }
1.369 + break;
1.370 + case EWsWinOpStoreDrawCommands:
1.371 + /* If the client asks us not to store commands, we still store the commands
1.372 + for the region of the window which can be seen through the parent, but
1.373 + won't attempt to obtain the entire window.
1.374 + */
1.375 + if (*aCmd.Bool)
1.376 + {
1.377 + SetScope(EStoreEntireWindow);
1.378 + }
1.379 + else
1.380 + {
1.381 + // Clients that turn their redraw store off will still get one,
1.382 + // but it will only attempt to store the current viewport.
1.383 + SetScope(EStoreViewport);
1.384 + }
1.385 + break;
1.386 + case EWsWinOpHandleTransparencyUpdate: // deprecated
1.387 + case EWsWinOpSetTransparencyBitmap: // deprecated
1.388 + case EWsWinOpSetTransparencyFactor: // deprecated
1.389 + case EWsWinOpSetTransparencyBitmapCWs: // deprecated
1.390 + break; // do nothing.
1.391 + case EWsWinOpIsRedrawStoreEnabled:
1.392 + SetReply(ETrue);
1.393 + break;
1.394 + case EWsWinOpClearRedrawStore:
1.395 + DiscardStoredCommands();
1.396 + break;
1.397 + case EWsWinOpSetBackgroundSurface:
1.398 + SetBackgroundSurfaceL(*aCmd.Surface);
1.399 + break;
1.400 + case EWsWinOpSetBackgroundSurfaceConfig:
1.401 + SetBackgroundSurfaceL(aCmd.SurfaceConfigurationAndTrigger->surfaceConfig, aCmd.SurfaceConfigurationAndTrigger->triggerRedraw, EFalse);
1.402 + break;
1.403 + case EWsWinOpRemoveBackgroundSurface:
1.404 + RemoveBackgroundSurface(*aCmd.Bool);
1.405 + break;
1.406 + case EWsWinOpGetBackgroundSurfaceConfig:
1.407 + {
1.408 + TSurfaceConfiguration tempConfiguration = *aCmd.SurfaceConfiguration;
1.409 + GetBackgroundSurfaceL(tempConfiguration);
1.410 + TInt tempSize = aCmd.SurfaceConfiguration->Size();
1.411 + if (sizeof(TSurfaceConfiguration)<tempSize)
1.412 + tempSize = sizeof(TSurfaceConfiguration);
1.413 + CWsClient::ReplyBuf(&tempConfiguration,tempSize);
1.414 + }
1.415 + break;
1.416 + default:
1.417 + return(EFalse);
1.418 + }
1.419 + return(ETrue);
1.420 + }
1.421 +
1.422 +/**
1.423 +*/
1.424 +void CWsRedrawMsgWindow::BeginRedraw(const TRect* aRect)
1.425 + {
1.426 + if(InRedraw())
1.427 + OwnerPanic(EWservPanicDrawCommandsInvalidState);
1.428 + iFlags|=EBeginEndRedraw;
1.429 + TRAPD(err,DoBeginRedrawL(aRect));
1.430 + DiscardStoredCommandsIfError(err);
1.431 + }
1.432 +
1.433 +void CWsRedrawMsgWindow::DoBeginRedrawL(const TRect* aRect)
1.434 + {
1.435 + const TRect redrawRect = (aRect ? *aRect : TRect(WsWin()->Size()));
1.436 + if (redrawRect.IsEmpty())
1.437 + {
1.438 + //Skip empty rects since they are not added to the region
1.439 + iCurrentSegment = NULL;
1.440 + }
1.441 + else
1.442 + {
1.443 + CreateNewSegmentL(redrawRect, CWsRedrawMsgWindow::ESegmentTypePendingRedraw);
1.444 + if (!iAtomic)
1.445 + PromotePendingSegment();
1.446 + }
1.447 + }
1.448 +
1.449 +void CWsRedrawMsgWindow::Invalidate(const TRect * aRect)
1.450 + {
1.451 + //The memory allocation in this function can trigger a call to ReleaseMemory(), which would
1.452 + //recursively call this function again. This would cause a memory leak. To avoid this
1.453 + //we call Lock() to block ReleaseMemory() from this object while executing this function.
1.454 + Lock();
1.455 + if (!aRect)
1.456 + {
1.457 + iInvalid.Clear();
1.458 + iInvalid.Copy(iWsWin->WindowArea());
1.459 + iInvalid.Offset(-iWsWin->Origin());
1.460 + }
1.461 + else if((!aRect->IsEmpty()) && aRect->IsNormalized())
1.462 + {
1.463 + iInvalid.AddRect(*aRect);
1.464 + iInvalid.Tidy();
1.465 + }
1.466 + if (iWsWin->IsVisible())
1.467 + {
1.468 + QueueRedraw();
1.469 + iWsWin->WsOwner()->TriggerRedraw(); //wtf isn't the redrawq already scheduling itself?
1.470 + }
1.471 + Unlock();
1.472 + }
1.473 +
1.474 +/**
1.475 +Obtains a region from the redraw store, intersects it with the global redraw region which
1.476 +we have been asked to draw to, and returns the intersection.
1.477 +*/
1.478 +const TRegion * CWsRedrawMsgWindow::ReadRegion(const TInt aRegionNum)
1.479 + {
1.480 + // We are drawing to the global region, and we have a region in the redraw store to clip to.
1.481 + // We want the intersection of these to actually draw to.
1.482 + iLocalRedrawRegion.Copy(iRedrawSegments[aRegionNum]->iRegion);
1.483 + iLocalRedrawRegion.Offset(WsWin()->Origin());
1.484 + iLocalRedrawRegion.Intersect(*iRedrawRegion);
1.485 + iLocalRedrawRegion.Tidy();
1.486 +
1.487 + // If the resulting region is empty there is no point drawing its corresponding commands
1.488 + if (iLocalRedrawRegion.IsEmpty())
1.489 + return NULL;
1.490 + else
1.491 + return &iLocalRedrawRegion;
1.492 + }
1.493 +
1.494 +void CWsRedrawMsgWindow::SubtractRectFromSegmentArray(const TRect& aRect)
1.495 + {
1.496 + for (TInt regionNum = iRedrawSegments.Count() - 1; regionNum >= 0; --regionNum)
1.497 + {
1.498 + if (iRedrawSegments[regionNum]->iRedrawSegmentType != ESegmentTypePendingRedraw)
1.499 + {
1.500 + RWsRegion& region = iRedrawSegments[regionNum]->iRegion;
1.501 + region.SubRect(aRect);
1.502 + if (region.CheckError())
1.503 + {
1.504 + // Ouch. Drop the now broken segment and ask for a full redraw
1.505 + delete iRedrawSegments[regionNum];
1.506 + iRedrawSegments.Remove(regionNum);
1.507 + Invalidate();
1.508 + }
1.509 + else
1.510 + {
1.511 + // check if region has zero uncovered rectangles left
1.512 + if (region.IsEmpty())
1.513 + { // delete draw commands, release bitmaps and fonts
1.514 + delete iRedrawSegments[regionNum];
1.515 + iRedrawSegments.Remove(regionNum);
1.516 + }
1.517 + else
1.518 + {
1.519 + if (region.Count() > KRegionCompressThreshold)
1.520 + { // tidy up the rectangles
1.521 + region.Tidy();
1.522 + }
1.523 + }
1.524 + }
1.525 + }
1.526 + }
1.527 + // coverity[extend_simple_error]
1.528 + }
1.529 +
1.530 +/*------------------------------------------------------------------------------
1.531 + Description: Clears out the command buffer if aError indicates an
1.532 + error has occurred whilst storing commands.
1.533 + -----------------------------------------------------------------------------*/
1.534 +void CWsRedrawMsgWindow::DiscardStoredCommandsIfError(TInt aError)
1.535 + {
1.536 + if (aError != KErrNone)
1.537 + {
1.538 + // Discard stored commands by clearing out the command buffer
1.539 + DiscardStoredCommands();
1.540 +
1.541 + if(!(iFlags&ECurrentRedrawFailedStorage)) //first time we fail during the current redraw
1.542 + {
1.543 + iFlags |= ECurrentRedrawFailedStorage;
1.544 +
1.545 + if(!(iFlags&EPreviousRedrawFailedStorage)) //unless the previous redraw failed as well (to avoid infinite loop)
1.546 + {
1.547 + Invalidate(); //request new redraw from the client
1.548 + }
1.549 + }
1.550 + }
1.551 + }
1.552 +
1.553 +/*------------------------------------------------------------------------------
1.554 + Description: If the graphics context has changed and we are currently storing
1.555 + commands, store the data given by aCmdData.
1.556 +
1.557 + -----------------------------------------------------------------------------*/
1.558 +TBool CWsRedrawMsgWindow::DrawCommand(CWsGc* aGc,const TAny *aCmdData)
1.559 + {
1.560 + // Store commands, clearing out buffer if error occurs.
1.561 + TRAPD(err,StoreDrawCommandL(aGc,aCmdData));
1.562 + DiscardStoredCommandsIfError(err);
1.563 + return EFalse;
1.564 + }
1.565 +
1.566 +void CWsRedrawMsgWindow::GcAttributeChange(CWsGc* aGc,const TAny *aCmdData)
1.567 + {
1.568 + if (iLastDrawGc == aGc && iCurrentSegment)
1.569 + {
1.570 + TInt err = KErrNone;
1.571 + if (IsWsFontOperation(CWsClient::iCurrentCommand.iOpcode))
1.572 + {
1.573 + TWsGcCmdUnion pData;
1.574 + pData.any=aCmdData;
1.575 + TRAP(err,AddWsFontL(*pData.UInt));
1.576 + }
1.577 + if (KErrNone == err)
1.578 + {
1.579 + TRAP(err,AppendCommandL(aCmdData));
1.580 + }
1.581 + DiscardStoredCommandsIfError(err);
1.582 + }
1.583 +
1.584 + // INC135845:
1.585 + // Retain the bitmap handle for the lifetime of the redraw store
1.586 + // If the client destroys it, we will still have a reference to it
1.587 + if (iCurrentSegment && CWsClient::iCurrentCommand.iOpcode == EWsGcOpUseBrushPattern)
1.588 + {
1.589 + TInt err = KErrNone;
1.590 + TWsGcCmdUnion pData;
1.591 + pData.any=aCmdData;
1.592 + TRAP(err, AddFbsBitmapsL(*pData.handle, 0));
1.593 + DiscardStoredCommandsIfError(err);
1.594 + }
1.595 + }
1.596 +
1.597 +void CWsRedrawMsgWindow::GcDeactivate(CWsGc* aGc)
1.598 + {
1.599 + if (iLastDrawGc==aGc)
1.600 + iLastDrawGc=NULL;
1.601 + }
1.602 +
1.603 +inline TBool CWsRedrawMsgWindow::IsFbsBitmapOperation(TInt aOpCode) const
1.604 + {
1.605 + WS_ASSERT_DEBUG((EWsGcOpGdiBlt3==EWsGcOpGdiBlt2+1)&&(EWsGcOpGdiBltMasked==EWsGcOpGdiBlt3+1)
1.606 + &&(EWsGcOpDrawBitmap2==EWsGcOpDrawBitmap+1)&&(EWsGcOpDrawBitmap3==EWsGcOpDrawBitmap2+1)&&(EWsGcOpDrawBitmapMasked==EWsGcOpDrawBitmap3+1),EWsPanicBitmapOpcodeInvalid);
1.607 + return (aOpCode>=EWsGcOpGdiBlt2&&aOpCode<=EWsGcOpGdiBltMasked)||(aOpCode>=EWsGcOpDrawBitmap&&aOpCode<=EWsGcOpDrawBitmapMasked)||(aOpCode==EWsGcOpGdiAlphaBlendBitmaps);
1.608 + }
1.609 +
1.610 +inline TBool CWsRedrawMsgWindow::IsWsBitmapOperation(TInt aOpCode) const
1.611 + {
1.612 + WS_ASSERT_DEBUG((EWsGcOpGdiBlt3==EWsGcOpGdiBlt2+1)&&(EWsGcOpGdiBltMasked==EWsGcOpGdiBlt3+1)
1.613 + &&(EWsGcOpDrawBitmap2==EWsGcOpDrawBitmap+1)&&(EWsGcOpDrawBitmap3==EWsGcOpDrawBitmap2+1)&&(EWsGcOpDrawBitmapMasked==EWsGcOpDrawBitmap3+1),EWsPanicBitmapOpcodeInvalid);
1.614 + return (aOpCode>=EWsGcOpGdiWsBlt2&&aOpCode<=EWsGcOpGdiWsBltMasked)||(aOpCode==EWsGcOpGdiWsAlphaBlendBitmaps)||(aOpCode==EWsGcOpWsDrawBitmapMasked);
1.615 + }
1.616 +
1.617 +inline TBool CWsRedrawMsgWindow::IsWsFontOperation(TInt aOpCode) const
1.618 + {
1.619 + return aOpCode==EWsGcOpUseFont;
1.620 + }
1.621 +
1.622 +inline TBool CWsRedrawMsgWindow::IsDrawWsGraphicOperation(TInt aOpCode) const
1.623 + {
1.624 + return (aOpCode == EWsGcOpDrawWsGraphic) || (aOpCode == EWsGcOpDrawWsGraphicPtr);
1.625 + }
1.626 +
1.627 +inline TBool CWsRedrawMsgWindow::IsWsDrawableSourceOperation(TInt aOpCode) const
1.628 + {
1.629 + return (aOpCode == EWsGcOpDrawResourceToPos) || (aOpCode == EWsGcOpDrawResourceToRect) || (aOpCode == EWsGcOpDrawResourceFromRectToRect) || (aOpCode == EWsGcOpDrawResourceWithData);
1.630 + }
1.631 +
1.632 +void CWsRedrawMsgWindow::ReplaceAndAppendCommandL(TInt aOpcode,const TAny* aCmdData)
1.633 + {
1.634 + const TInt KCharWidthInBytes = 2; // # of bytes for a Unicode character
1.635 + CWsClient* owner=iWsWin->WsOwner();
1.636 + WS_ASSERT_DEBUG(owner,EWsPanicDrawCommandsNullSession);
1.637 +
1.638 + // aCmdData doesn't contain data, it should be retrieved from client space using remote read
1.639 + TWsGcCmdUnion cmd;
1.640 + cmd.any=aCmdData;
1.641 + TUint16 newOpcode=EWsGcOpDrawText;
1.642 + TInt strLen=0;
1.643 + switch (aOpcode)
1.644 + {
1.645 + case EWsGcOpDrawTextPtr:
1.646 + newOpcode=EWsGcOpDrawText;
1.647 + strLen=cmd.DrawText->length;
1.648 + break;
1.649 + case EWsGcOpDrawTextVerticalPtr:
1.650 + newOpcode=EWsGcOpDrawTextVertical;
1.651 + strLen=cmd.DrawTextVertical->length;
1.652 + break;
1.653 + case EWsGcOpDrawBoxTextPtr:
1.654 + newOpcode=EWsGcOpDrawBoxText;
1.655 + strLen=cmd.BoxText->length;
1.656 + break;
1.657 + case EWsGcOpDrawBoxTextVerticalPtr:
1.658 + newOpcode=EWsGcOpDrawBoxTextVertical;
1.659 + strLen=cmd.DrawBoxTextVertical->length;
1.660 + break;
1.661 + case EWsGcOpDrawTextInContextPtr:
1.662 + newOpcode=EWsGcOpDrawTextInContextPtr1;
1.663 + strLen=cmd.DrawTextInContext->length;
1.664 + break;
1.665 + case EWsGcOpDrawTextInContextVerticalPtr:
1.666 + newOpcode=EWsGcOpDrawTextInContextVerticalPtr1;
1.667 + strLen=cmd.DrawTextInContextVertical->length;
1.668 + break;
1.669 + case EWsGcOpDrawBoxTextInContextPtr:
1.670 + newOpcode=EWsGcOpDrawBoxTextInContextPtr1;
1.671 + strLen=cmd.BoxTextInContext->length;
1.672 + break;
1.673 + case EWsGcOpDrawBoxTextInContextVerticalPtr:
1.674 + newOpcode=EWsGcOpDrawBoxTextInContextVerticalPtr1;
1.675 + strLen=cmd.DrawBoxTextInContextVertical->length;
1.676 + break;
1.677 + }
1.678 + TInt strSize = strLen * KCharWidthInBytes;
1.679 + TInt oldCmdLen=CWsClient::iCurrentCommand.iCmdLength;
1.680 + TInt newCmdLen=sizeof(TWsCmdHeaderBase)+oldCmdLen+strSize;
1.681 + // resize buffer
1.682 + ExpandCommandBufferL(newCmdLen);
1.683 + // update current command to reflect the new command and data
1.684 + CWsClient::iCurrentCommand.iOpcode=newOpcode;
1.685 + CWsClient::iCurrentCommand.iOpcode|=EWsGcOpFlagDrawOp;
1.686 + CWsClient::iCurrentCommand.iCmdLength=(TInt16)(oldCmdLen+strSize);
1.687 + // write command header
1.688 + CommandBufferWrite(&CWsClient::iCurrentCommand, sizeof(TWsCmdHeaderBase));
1.689 + // write command
1.690 + CommandBufferWrite(aCmdData, oldCmdLen);
1.691 +
1.692 + // remote read
1.693 + TBuf<KReadBufferMaxLen> buf;
1.694 + TInt len=KReadBufferMaxLen;
1.695 + TInt bufOffset=0;
1.696 + TInt toGo=strLen;
1.697 + while(toGo>0)
1.698 + {
1.699 + if (len>toGo)
1.700 + len=toGo;
1.701 + owner->RemoteRead(buf,bufOffset);
1.702 + CommandBufferWrite(buf.Ptr(), len * KCharWidthInBytes);
1.703 + bufOffset+=len;
1.704 + toGo-=len;
1.705 + }
1.706 + }
1.707 +
1.708 +/*------------------------------------------------------------------------------
1.709 + Description: Stores drawing related commands into the command buffer
1.710 + -----------------------------------------------------------------------------*/
1.711 +void CWsRedrawMsgWindow::StoreDrawCommandL(CWsGc* aGc,const TAny *aCmdData)
1.712 + {
1.713 + TWsGcOpcodes currentOpcode = static_cast<TWsGcOpcodes>(CWsClient::iCurrentCommand.iOpcode);
1.714 +
1.715 + if (!InRedraw())
1.716 + {
1.717 + // We've received a drawing operation outside of a BeginRedraw()/EndRedraw()
1.718 + // bracketed block. This is known as a "non redraw drawing operation".
1.719 +
1.720 +#ifdef __WINS__
1.721 + // If configured, panic on non redraw drawing operations.
1.722 + if( CWsClient::DebugEnforceRedrawCallingConvention())
1.723 + CWsClient::PanicCurrentClient(EWservPanicWindowBeginRedrawNotCalled);
1.724 +#endif
1.725 +
1.726 + // We get here if we are not configured to panic when a non-redraw drawing operation
1.727 + // is issued.
1.728 + // We ignore this DrawOp as WServ2 is not supporting non-redraw drawing because
1.729 + // storing such non-redraw commands in the redraw store consumes significant
1.730 + // amounts of memory, and reduces rendering performance.
1.731 + //
1.732 + // Issuing non-redraw drawing from a client Redrawer is a special case. Here
1.733 + // an infinite loop would occur if we mark the area as invalid. Therefore
1.734 + // we must always panic in this case.
1.735 + if (iWsWin->WsOwner()->ClientProcessingRedrawEvent())
1.736 + {
1.737 + // Non-redraw drawing in the Redrawer!!
1.738 + CWsClient::PanicCurrentClient(EWservPanicWindowBeginRedrawNotCalled);
1.739 + }
1.740 + else
1.741 + {
1.742 + // The Redrawer will eventually fix up the screen
1.743 + Invalidate();
1.744 + }
1.745 + return;
1.746 + }
1.747 +
1.748 + // If there is no current segment then we have discarded it at some point
1.749 + // since beginning this redraw.
1.750 + if (iCurrentSegment)
1.751 + {
1.752 + TWsGcCmdUnion pData;
1.753 + pData.any = aCmdData;
1.754 + if (IsFbsBitmapOperation(currentOpcode))
1.755 + {
1.756 + TInt maskHandle = 0;
1.757 + TInt handle = aGc->FbsBitmapHandle(currentOpcode, pData, maskHandle);
1.758 + AddFbsBitmapsL(handle, maskHandle);
1.759 + }
1.760 + else if (IsWsBitmapOperation(currentOpcode))
1.761 + {
1.762 + TInt maskHandle = 0;
1.763 + TInt handle = aGc->WsBitmapHandle(currentOpcode, pData, maskHandle);
1.764 + AddWsBitmapsL(handle, maskHandle);
1.765 + }
1.766 + else if (IsDrawWsGraphicOperation(currentOpcode))
1.767 + {
1.768 + TGraphicDrawerId drawerId;
1.769 + drawerId.iId = pData.WsGraphic->iId;
1.770 + drawerId.iIsUid = (pData.WsGraphic->iFlags & EWsGraphicIdUid);
1.771 + iCurrentSegment->AddDrawerL(drawerId);
1.772 + }
1.773 + else if (IsWsDrawableSourceOperation(currentOpcode))
1.774 + {
1.775 + TInt handle = aGc->WsDrawableSourceHandle(currentOpcode, pData);
1.776 + AddWsDrawableSourceL(handle);
1.777 + }
1.778 +
1.779 + // If the graphics context has changed since last time store the new graphics
1.780 + // context attributes.
1.781 + if (aGc != iLastDrawGc)
1.782 + {
1.783 + StoreAllGcAttributesL(aGc);
1.784 + iLastDrawGc = aGc;
1.785 + }
1.786 +
1.787 + // For operation which requires remote read from client space, we must retrieve that data and store
1.788 + // it in command buffer at server side and change opcode if necessary e.g EWsGcOpDrawTextPtr to EWsGcOpDrawText
1.789 + // to avoid remote read during DoDrawing operation
1.790 + if (IsRemoteReadRequired(currentOpcode))
1.791 + ReplaceAndAppendCommandL(currentOpcode,aCmdData);
1.792 + else
1.793 + // Append the command data to the command buffer
1.794 + AppendCommandL(aCmdData, EWsGcOpFlagDrawOp);
1.795 + }
1.796 + }
1.797 +
1.798 +/*------------------------------------------------------------------------------
1.799 + Description: Stores given drawing command data into the command buffer.
1.800 + -----------------------------------------------------------------------------*/
1.801 +void CWsRedrawMsgWindow::AppendCommandL(const TAny* aCmdData, const TUint16 aOpcodeFlags)
1.802 + {
1.803 + if (CWsClient::iCurrentCommand.iOpcode == EWsGcOpSetClippingRegion)
1.804 + {
1.805 + // The client is defining a clipping region
1.806 +
1.807 + // make room for the header
1.808 + ExpandCommandBufferL(sizeof(TWsCmdHeaderBase));
1.809 +
1.810 + // Externalize the clipping region data from position after the header
1.811 + RBufWriteStream bufWriteStream;
1.812 + bufWriteStream.Open(*CurrentDrawCommandBuffer(), CurrentCommandBufferWritePos() + sizeof(TWsCmdHeaderBase));
1.813 + CleanupClosePushL(bufWriteStream);
1.814 + TInt dataLen = iLastDrawGc->ExternalizeClippingRegionL(bufWriteStream);
1.815 +
1.816 + // Setup the clipping region data header
1.817 + CWsClient::iCurrentCommand.iOpcode = EWsStoreClippingRegion;
1.818 + CWsClient::iCurrentCommand.iCmdLength = REINTERPRET_CAST(TInt16&,dataLen);
1.819 +
1.820 + // Store command header for clipping region data at current write position
1.821 + CommandBufferWrite(&CWsClient::iCurrentCommand,sizeof(TWsCmdHeaderBase));
1.822 +
1.823 + // Update write position for command data
1.824 + iCurrentSegment->iCurrentCommandBufferWritePos += dataLen;
1.825 + CleanupStack::PopAndDestroy(&bufWriteStream);
1.826 + }
1.827 + else
1.828 + {
1.829 + TUint16 opcode = CWsClient::iCurrentCommand.iOpcode;
1.830 + CWsClient::iCurrentCommand.iOpcode |= aOpcodeFlags;
1.831 +
1.832 + // ensure room in command buffer
1.833 + ExpandCommandBufferL(sizeof(TWsCmdHeaderBase) + CWsClient::iCurrentCommand.iCmdLength);
1.834 +
1.835 + // Store command header to current position
1.836 + CommandBufferWrite(&CWsClient::iCurrentCommand, sizeof(TWsCmdHeaderBase));
1.837 +
1.838 + // If there's command data (other than header), store it
1.839 + if (CWsClient::iCurrentCommand.iCmdLength > 0)
1.840 + {
1.841 + CommandBufferWrite(aCmdData, CWsClient::iCurrentCommand.iCmdLength);
1.842 + }
1.843 +
1.844 + CWsClient::iCurrentCommand.iOpcode = opcode;
1.845 + }
1.846 + }
1.847 +
1.848 +
1.849 +/*------------------------------------------------------------------------------
1.850 + Description: Stores graphics context information into the command buffer
1.851 + from the current write position.
1.852 + -----------------------------------------------------------------------------*/
1.853 +void CWsRedrawMsgWindow::StoreAllGcAttributesL(CWsGc* aGc)
1.854 + {
1.855 + // In order for the externalize below to work correctly from
1.856 + // a non-zero position we have to create the header placeholder
1.857 + ExpandCommandBufferL(sizeof(TWsCmdHeaderBase));
1.858 +
1.859 + // Externalise GC attribute data. We do this before writing the
1.860 + // header as we do not know the size of the data yet and it is
1.861 + // part of the header.
1.862 + TInt numOfBytesAdded = aGc->ExternalizeL(*CurrentDrawCommandBuffer(),
1.863 + CurrentCommandBufferWritePos() + sizeof(TWsCmdHeaderBase));
1.864 +
1.865 + // Setup the header
1.866 + TWsCmdHeaderBase cmdHeader;
1.867 + cmdHeader.iCmdLength = (TInt16) numOfBytesAdded; // as calculated above
1.868 + cmdHeader.iOpcode = (TInt16) EWsStoreAllGcAttributes;
1.869 +
1.870 + // Store the header for the GC data into the space we created
1.871 + CommandBufferWrite(&cmdHeader, sizeof(TWsCmdHeaderBase));
1.872 +
1.873 + // Update write position for command data
1.874 + iCurrentSegment->iCurrentCommandBufferWritePos += numOfBytesAdded;
1.875 + }
1.876 +
1.877 +LOCAL_C void CallSegmentEnd(TAny* aAnnotationObserver)
1.878 + {
1.879 + static_cast<MWsDrawAnnotationObserver*>(aAnnotationObserver)->SegmentRedrawEnd();
1.880 + }
1.881 +
1.882 +/*------------------------------------------------------------------------------
1.883 + Description: Loops through the whole of the current command buffer, processing
1.884 + each in turn.
1.885 + -----------------------------------------------------------------------------*/
1.886 +void CWsRedrawMsgWindow::DrawCommandsL()
1.887 + {
1.888 + LOG_WINDOW_DRAW_COMMANDS_START(WsWin(), iRedrawRegion);
1.889 + WS_ASSERT_DEBUG(iMemoryLock > 0, EWsPanicMemoryLock);
1.890 + static TBuf8<EClientBufferMaxSize> buf;
1.891 + TInt regionCount = iRedrawSegments.Count();
1.892 +
1.893 + for (TInt regionNum = 0; regionNum < regionCount; ++regionNum)
1.894 + {
1.895 + CRedrawSegment* segment = iRedrawSegments[regionNum];
1.896 + LOG_REDRAW_SEGMENT(regionNum, segment->iRedrawSegmentType);
1.897 + if (segment->iRedrawSegmentType == ESegmentTypePendingRedraw)
1.898 + continue;
1.899 +
1.900 + // The amount of commands we process is given by the value of the
1.901 + // current write position rather than the size of the command buffer.
1.902 + // Note: the write position is incremented as each command is stored and
1.903 + // will typically be less than the buffer size.
1.904 + const TInt length = segment->iCurrentCommandBufferWritePos;
1.905 +
1.906 + // need to draw this region?
1.907 + const TRegion * localDrawRegion = 0;
1.908 + if (length)
1.909 + localDrawRegion = ReadRegion(regionNum);
1.910 + if (localDrawRegion)
1.911 + {
1.912 + MWsDrawAnnotationObserver* const annoObs = Screen()->DrawAnnotationObserver();
1.913 + if(annoObs)
1.914 + {
1.915 + annoObs->SegmentRedrawStart(*localDrawRegion);
1.916 + CleanupStack::PushL(TCleanupItem(CallSegmentEnd, annoObs));
1.917 + }
1.918 + CPlaybackGc::Instance()->SetTargetRegion(localDrawRegion);
1.919 + LOG_REDRAW_SEGMENT_REGION(localDrawRegion)
1.920 +
1.921 + TWsCmdHeaderBase header;
1.922 + TInt pos = 0; // Set to first command position in buffer
1.923 + CBufSeg* drawCmdBuffer = segment->iDrawCommands;
1.924 +
1.925 +#ifdef _DEBUG
1.926 + // Read the first command header. The associated opcode must always be
1.927 + // EWsStoreAllGcAttributes as this is always the first stored item.
1.928 + drawCmdBuffer->Read(pos,&header,sizeof(TWsCmdHeaderBase));
1.929 + WS_ASSERT_DEBUG(header.iOpcode == EWsStoreAllGcAttributes, EWsPanicDrawCommandsBufferCorrupt);
1.930 +#endif
1.931 +
1.932 + // Read through remaining commands
1.933 + while (pos < length)
1.934 + {
1.935 + // Get header of command
1.936 + drawCmdBuffer->Read(pos, &header, sizeof(TWsCmdHeaderBase));
1.937 + pos += sizeof(TWsCmdHeaderBase);
1.938 +
1.939 + switch(header.iOpcode)
1.940 + {
1.941 + case EWsStoreAllGcAttributes:
1.942 + {
1.943 + // Header indicates command encapsulates gc data
1.944 + CPlaybackGc::Instance()->Reset();
1.945 +
1.946 + // Read gc data
1.947 + CPlaybackGc::Instance()->InternalizeL(*drawCmdBuffer,pos);
1.948 +
1.949 + }
1.950 + break;
1.951 + case EWsStoreClippingRegion:
1.952 + {
1.953 + // Clipping region data read in from current position via stream
1.954 + RBufReadStream bufReadStream;
1.955 + bufReadStream.Open(*drawCmdBuffer,pos);
1.956 + CleanupClosePushL(bufReadStream);
1.957 + CPlaybackGc::Instance()->InternalizeClippingRegionL(bufReadStream);
1.958 + CleanupStack::PopAndDestroy(&bufReadStream);
1.959 + }
1.960 + break;
1.961 + default:
1.962 + {
1.963 + // Another type of command. Read it.
1.964 + CWsClient::iCurrentCommand.iCmdLength = header.iCmdLength;
1.965 + drawCmdBuffer->Read(pos,buf,header.iCmdLength);
1.966 +
1.967 + TInt opcode = header.iOpcode;
1.968 +
1.969 + // Drawing command?
1.970 + if (opcode & EWsGcOpFlagDrawOp)
1.971 + {
1.972 + opcode &= ~EWsGcOpFlagDrawOp;
1.973 + }
1.974 + if (opcode > -1)
1.975 + {
1.976 + LOG_PLAYBACK_GC_COMMAND(opcode, buf.Ptr())
1.977 + CPlaybackGc::Instance()->CommandL(static_cast<TWsGcOpcodes>(opcode),buf);
1.978 + }
1.979 + }
1.980 + break;
1.981 + }
1.982 + pos += header.iCmdLength; // Move on, header indicates length
1.983 + }
1.984 + if (annoObs)
1.985 + CleanupStack::PopAndDestroy(annoObs);
1.986 + }
1.987 + }
1.988 + LOG_WINDOW_DRAW_COMMANDS_END(WsWin());
1.989 + }
1.990 +
1.991 +/*------------------------------------------------------------------------------
1.992 + Description: Called when the currently stored graphics commands
1.993 + are no longer required.
1.994 + -----------------------------------------------------------------------------*/
1.995 +void CWsRedrawMsgWindow::DiscardStoredCommands()
1.996 + {
1.997 + iCurrentSegment = NULL;
1.998 + if (iRedrawSegments.Count() > 0)
1.999 + {
1.1000 + // First of all, if we have any redraws pending, update the screen with
1.1001 + // whatever commands we have before we throw them away:
1.1002 + if (iFlags & EPendingScheduledDraw)
1.1003 + Screen()->DoRedrawNow();
1.1004 +
1.1005 + // for all regions or just Partial Redraw regions > index 0: delete bitmaps and draw commands
1.1006 + iRedrawSegments.ResetAndDestroy();
1.1007 +
1.1008 + iLastDrawGc = NULL;
1.1009 + }
1.1010 + }
1.1011 +
1.1012 +void CWsRedrawMsgWindow::CreateNewSegmentL(const TRect& aRect, TRedrawSegmentType aNewRedrawRegionType)
1.1013 + {
1.1014 + CWsRedrawMsgWindow::CRedrawSegment* newRegion = CWsRedrawMsgWindow::CRedrawSegment::NewLC(aRect, aNewRedrawRegionType, *this);
1.1015 +
1.1016 + iRedrawSegments.AppendL(newRegion);
1.1017 + iCurrentSegment = newRegion;
1.1018 + CleanupStack::Pop(newRegion);
1.1019 +
1.1020 + // Set iLastDrawGc to NULL. This will cause all GC attributes to be stored
1.1021 + // in redraw store when the window receives the next command
1.1022 + iLastDrawGc = NULL;
1.1023 + }
1.1024 +
1.1025 +void CWsRedrawMsgWindow::AddFbsBitmapsL(TInt aHandle, TInt aMaskHandle)
1.1026 + {
1.1027 + AddFbsBitmapRefL(aHandle);
1.1028 + if (aMaskHandle)
1.1029 + {
1.1030 + AddFbsBitmapRefL(aMaskHandle);
1.1031 + }
1.1032 + }
1.1033 +
1.1034 +void CWsRedrawMsgWindow::AddFbsBitmapRefL(TInt aHandle)
1.1035 + {
1.1036 + TInt indexBitmapRef = iFbsBitmapRefArray.FindInOrder(aHandle, &FindBitmapRefByHandle);
1.1037 +
1.1038 + // check whether the window already has a reference for the bitmap through a segment other than the current one
1.1039 + if (indexBitmapRef >=0)
1.1040 + {
1.1041 + TBool handleAppend = iCurrentSegment->AppendNonDuplicateBitmapHandleL(aHandle);
1.1042 + if (handleAppend)
1.1043 + {
1.1044 + iFbsBitmapRefArray[indexBitmapRef]->IncRefCount();
1.1045 + }
1.1046 + }
1.1047 + else
1.1048 + {
1.1049 + CFbsBitmapRef* bitmapRef = new(ELeave) CFbsBitmapRef;
1.1050 + CleanupStack::PushL(bitmapRef);
1.1051 + if (bitmapRef->Duplicate(aHandle)!=KErrNone)
1.1052 + OwnerPanic(EWservPanicBitmap);
1.1053 + iFbsBitmapRefArray.InsertInOrderL(bitmapRef, TLinearOrder<CFbsBitmapRef>(InsertBitmapRefByHandle));
1.1054 + CleanupStack::Pop(bitmapRef);
1.1055 + bitmapRef->IncRefCount();
1.1056 +#ifdef _DEBUG
1.1057 + TBool bitmapHandleAppended = EFalse;
1.1058 + TRAPD(err, bitmapHandleAppended = iCurrentSegment->AppendNonDuplicateBitmapHandleL(aHandle);
1.1059 + WS_ASSERT_DEBUG(bitmapHandleAppended,EWsPanicUnexpectedBitmapHandleInArray););
1.1060 +#else
1.1061 + TRAPD(err, iCurrentSegment->AppendNonDuplicateBitmapHandleL(aHandle););
1.1062 +#endif
1.1063 + if (err != KErrNone)
1.1064 + {
1.1065 + // Cleanup the array, then propagate the error
1.1066 + indexBitmapRef = iFbsBitmapRefArray.FindInOrder(aHandle, &FindBitmapRefByHandle);
1.1067 + if (indexBitmapRef >= 0)
1.1068 + {
1.1069 + iFbsBitmapRefArray[indexBitmapRef]->DecRefCount();
1.1070 + delete iFbsBitmapRefArray[indexBitmapRef];
1.1071 + iFbsBitmapRefArray.Remove(indexBitmapRef);
1.1072 + }
1.1073 + User::Leave(err);
1.1074 + }
1.1075 + }
1.1076 + }
1.1077 +
1.1078 +// Only called during playback of a redraw store segment
1.1079 +CFbsBitmap* CWsRedrawMsgWindow::BitmapFromHandle(TInt aHandle) const
1.1080 + {
1.1081 + const TInt index = iFbsBitmapRefArray.FindInOrder(aHandle, &FindBitmapRefByHandle);
1.1082 + if (index != KErrNotFound)
1.1083 + {
1.1084 + CFbsBitmap* bitmap = iFbsBitmapRefArray[index];
1.1085 + return bitmap;
1.1086 + }
1.1087 + return NULL;
1.1088 + }
1.1089 +
1.1090 +
1.1091 +void CWsRedrawMsgWindow::AddWsBitmapsL(TInt aHandle, TInt aMaskHandle)
1.1092 + {
1.1093 + if (iWsWin->WsOwner() == NULL)
1.1094 + Panic(EWsPanicDrawCommandsInvalidState);
1.1095 + DWsBitmap * bmp = static_cast<DWsBitmap*>(iWsWin->WsOwner()->HandleToObj(aHandle, WS_HANDLE_BITMAP));
1.1096 + if (!bmp)
1.1097 + OwnerPanic(EWservPanicBitmap);
1.1098 + iCurrentSegment->AddWsBitmapL(bmp);
1.1099 + if (aMaskHandle)
1.1100 + {
1.1101 + bmp = static_cast<DWsBitmap*>(iWsWin->WsOwner()->HandleToObj(aMaskHandle, WS_HANDLE_BITMAP));
1.1102 + if (!bmp)
1.1103 + OwnerPanic(EWservPanicBitmap);
1.1104 + iCurrentSegment->AddWsBitmapL(bmp);
1.1105 + }
1.1106 + }
1.1107 +
1.1108 +void CWsRedrawMsgWindow::CRedrawSegment::AddWsBitmapL(DWsBitmap* bitmap)
1.1109 + {
1.1110 + iWsBitmapArray.AppendL(bitmap);
1.1111 + bitmap->IncRefCount();
1.1112 + }
1.1113 +
1.1114 +void CWsRedrawMsgWindow::AddWsFontL(TInt aHandle)
1.1115 + {
1.1116 + if (iWsWin->WsOwner()==NULL)
1.1117 + Panic(EWsPanicDrawCommandsInvalidState);
1.1118 + TDblQueIter<CWsFbsFont> iter(CWsFontCache::List());
1.1119 + CWsFbsFont* font=NULL;
1.1120 + while((font=iter++)!=NULL)
1.1121 + {
1.1122 + if (font->Handle()==aHandle)
1.1123 + break;
1.1124 + }
1.1125 + if (font)
1.1126 + {
1.1127 + iCurrentSegment->iWsFontArray.AppendL(font);
1.1128 + ++(font->iCount);
1.1129 + }
1.1130 + }
1.1131 +
1.1132 +void CWsRedrawMsgWindow::CRedrawSegment::AddDrawerL(TGraphicDrawerId aDrawerId)
1.1133 + {
1.1134 + TInt error = iDrawerArray.InsertInOrder(aDrawerId, TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare));
1.1135 + if (error != KErrAlreadyExists && error != KErrNone)
1.1136 + {
1.1137 + User::Leave(error);
1.1138 + }
1.1139 + }
1.1140 +
1.1141 +TBool CWsRedrawMsgWindow::CRedrawSegment::ContainsDrawers(const TArray<TGraphicDrawerId>& aDrawers,const TRegion& aRegion) const
1.1142 + {
1.1143 + TBool result = EFalse;
1.1144 + if (iDrawerArray.Count() > 0)
1.1145 + {
1.1146 + STACK_REGION tempRegion;
1.1147 + tempRegion.Intersection(iRegion, aRegion);
1.1148 + if (tempRegion.CheckError() || (tempRegion.Count() > 0) )
1.1149 + { // regions do intersect, (presumed if region had an error); so check for a matching Id
1.1150 + const TInt drawersCount = aDrawers.Count();
1.1151 + for (TInt idx = 0; idx < drawersCount; ++idx)
1.1152 + { // (iDrawerArray is kept sorted)
1.1153 + if (KErrNotFound != iDrawerArray.FindInOrder(aDrawers[idx], TLinearOrder<TGraphicDrawerId>(TGraphicDrawerId::Compare)))
1.1154 + {
1.1155 + result = ETrue;
1.1156 + break;
1.1157 + }
1.1158 +
1.1159 + const TInt count = iDrawerArray.Count();
1.1160 + for(TInt i = 0; i < count; i++)
1.1161 + {
1.1162 + const CWsGraphicDrawer* drawer = CWsTop::WindowServer()->ResolveGraphic(iDrawerArray[i]);
1.1163 + if(drawer && drawer->Contains(aDrawers))
1.1164 + {
1.1165 + result = ETrue;
1.1166 + break;
1.1167 + }
1.1168 + }
1.1169 + }
1.1170 + }
1.1171 + tempRegion.Close();
1.1172 + }
1.1173 + return result;
1.1174 + }
1.1175 +
1.1176 +void CWsRedrawMsgWindow::AddWsDrawableSourceL(TInt aHandle)
1.1177 + {
1.1178 + if (iWsWin->WsOwner() == NULL)
1.1179 + Panic(EWsPanicDrawCommandsInvalidState);
1.1180 + CWsDrawableSource* drawableSource = static_cast<CWsDrawableSource*>(iWsWin->WsOwner()->HandleToObj(aHandle, WS_HANDLE_DRAWABLE_SOURCE));
1.1181 + if (!drawableSource)
1.1182 + OwnerPanic(EWservPanicDrawableSource);
1.1183 + iCurrentSegment->AddWsDrawableSourceL(drawableSource);
1.1184 + }
1.1185 +
1.1186 +void CWsRedrawMsgWindow::CRedrawSegment::AddWsDrawableSourceL(CWsDrawableSource* aDrawableSource)
1.1187 + {
1.1188 + iWsDrawableSourceArray.AppendL(aDrawableSource);
1.1189 + aDrawableSource->IncRefCount();
1.1190 + }
1.1191 +
1.1192 +inline TBool CWsRedrawMsgWindow::NoBuffer() const
1.1193 + {
1.1194 + return (iRedrawSegments.Count() == 0);
1.1195 + }
1.1196 +
1.1197 +void CWsRedrawMsgWindow::ClientExposing()
1.1198 + {
1.1199 + Invalidate();
1.1200 + }
1.1201 +
1.1202 +/*------------------------------------------------------------------------------
1.1203 + Description: If a complete set of drawing commands have been stored
1.1204 + this method attempts to draw ALL the commands via DrawCommandsL().
1.1205 + It also draws the window in the background colour if the window is
1.1206 + opaque.
1.1207 + -----------------------------------------------------------------------------*/
1.1208 +void CWsRedrawMsgWindow::DrawWindow()
1.1209 + {
1.1210 + iFlags &= ~EPendingScheduledDraw;
1.1211 + // This is a happy window - it can draw itself whenever we ask.
1.1212 + CScreen* screen = Screen();
1.1213 + if ((screen->AutoClear() && (iFlags & EBackgroundClear)) || HasElement())
1.1214 + {
1.1215 + DrawBackgroundColor(*iRedrawRegion,(iFlags&EBackgroundClear)!=0);
1.1216 + }
1.1217 + if (HasElement())
1.1218 + {
1.1219 + screen->WindowElements().UnassignPlacedElements(*iRedrawRegion,*CliWin(),CPlaybackGc::Instance()->GcDrawingCount());
1.1220 + }
1.1221 + // If valid commands have been stored, draw them.
1.1222 + if (iRedrawSegments.Count() > 0)
1.1223 + {
1.1224 + Lock();
1.1225 + TRAP_IGNORE(DrawCommandsL());
1.1226 + Unlock();
1.1227 + }
1.1228 + if (HasElement()) //HasElement won't be cleared following first call above. May get set...
1.1229 + {
1.1230 + TInt state=screen->WindowElements().CleanUpPlacedElements(*CliWin(),CPlaybackGc::Instance()->GcDrawingCount());
1.1231 + if (state&CWindowElement::EFastPath)
1.1232 + {
1.1233 + if (HasElement())
1.1234 + {
1.1235 + screen->ElementAdded();
1.1236 + }
1.1237 + else
1.1238 + {
1.1239 + screen->ElementRemoved();
1.1240 + }
1.1241 + }
1.1242 + }
1.1243 + }
1.1244 +
1.1245 +void CWsRedrawMsgWindow::RemoveFromRedrawQueueIfEmpty()
1.1246 + {
1.1247 + if (iInvalid.Count()==0)
1.1248 + {
1.1249 + iInvalid.Clear(); // Ensures heap cell is freed, otherwise may be left as an empty cell
1.1250 + iWsWin->WsOwner()->RedrawQueue()->RemoveInvalid(this);
1.1251 + }
1.1252 + }
1.1253 +
1.1254 +const TRegion& CWsRedrawMsgWindow::InvalidArea() const
1.1255 + {
1.1256 + return(iInvalid);
1.1257 + }
1.1258 +
1.1259 +TBool CWsRedrawMsgWindow::NeedsRedraw() const
1.1260 +// If iInvalid has an persistant error it will not be reported as needing a redraw,
1.1261 +// this is needed as otherwise cases where validation of a window results
1.1262 +// in iInvalid having an error will get into an endless cycle of redraws.
1.1263 +// The down side of this is that sometimes a window will not be sent a redraw
1.1264 +// message when it needs it, some things can't be perfect!
1.1265 +//
1.1266 + {
1.1267 + if ((!iWsWin->IsVisible()) || iInvalid.IsEmpty())
1.1268 + return EFalse;
1.1269 +
1.1270 + TRect nextRedrawRect;
1.1271 + return GetRedrawRect(nextRedrawRect);
1.1272 + }
1.1273 +
1.1274 +TBool CWsRedrawMsgWindow::GetRedrawRect(TRect &aRect) const
1.1275 + {
1.1276 + if (iWsWin->ClientSetInvisible())
1.1277 + return EFalse;
1.1278 +
1.1279 + if(InRedraw())
1.1280 + {
1.1281 + aRect = iRedrawRect;
1.1282 + return (!aRect.IsEmpty());
1.1283 + }
1.1284 + else if(iInvalid.CheckError())
1.1285 + {
1.1286 + if (Screen()->ChangeTracking())
1.1287 + {
1.1288 + aRect = iWsWin->AbsRect();
1.1289 + }
1.1290 + else
1.1291 + {
1.1292 + if (iFlags & EStoringEntireWindow || iWsWin->VisibleRegion().CheckError())
1.1293 + {
1.1294 + aRect = iWsWin->AbsRect();
1.1295 + }
1.1296 + else
1.1297 + {
1.1298 + aRect = iWsWin->VisibleRegion().BoundingRect();
1.1299 + }
1.1300 + }
1.1301 + if (!(iFlags & EStoringEntireWindow))
1.1302 + iWsWin->ClipRectToViewport(aRect);
1.1303 + aRect.Move(-iWsWin->Origin());
1.1304 + return (!aRect.IsEmpty());
1.1305 + }
1.1306 + else if(iInvalid.Count())
1.1307 + {
1.1308 + if (iFlags & EStoringEntireWindow)
1.1309 + {
1.1310 + aRect = iInvalid.BoundingRect();
1.1311 + }
1.1312 + else
1.1313 + {
1.1314 + RWsRegion region;
1.1315 + region.Copy(iInvalid);
1.1316 + region.Offset(iWsWin->Origin());
1.1317 + if (!Screen()->ChangeTracking())
1.1318 + {
1.1319 + region.Intersect(iWsWin->VisibleRegion());
1.1320 + }
1.1321 + if (region.CheckError())
1.1322 + {
1.1323 + aRect = iInvalid.BoundingRect();
1.1324 + aRect.Move(iWsWin->Origin());
1.1325 + }
1.1326 + else
1.1327 + {
1.1328 + aRect = region.BoundingRect();
1.1329 + }
1.1330 + region.Close();
1.1331 + iWsWin->ClipRectToViewport(aRect);
1.1332 + aRect.Move(-iWsWin->Origin());
1.1333 + }
1.1334 + return (!aRect.IsEmpty());
1.1335 + }
1.1336 + else
1.1337 + {
1.1338 + return EFalse;
1.1339 + }
1.1340 + }
1.1341 +
1.1342 +const TRegion &CWsRedrawMsgWindow::BaseDrawRegion() const
1.1343 + {
1.1344 + return(iWsWin->VisibleRegion());
1.1345 + }
1.1346 +
1.1347 +void CWsRedrawMsgWindow::ClipInvalidRegion(const TRect &aRect)
1.1348 + {
1.1349 + if (iInvalid.Count()>0)
1.1350 + {
1.1351 + iInvalid.ClipRect(aRect);
1.1352 + RemoveFromRedrawQueueIfEmpty();
1.1353 + }
1.1354 + }
1.1355 +
1.1356 +void CWsRedrawMsgWindow::EndRedraw()
1.1357 + {
1.1358 + if(!InRedraw())
1.1359 + OwnerPanic(EWservPanicDrawCommandsInvalidState);
1.1360 + if (iCurrentSegment)
1.1361 + {
1.1362 + iCurrentSegment->iDrawCommands->Compress();
1.1363 + if (iAtomic)
1.1364 + PromotePendingSegment();
1.1365 +
1.1366 + // Schedule an update of the area of the screen we just drew to:
1.1367 + iFlags |= EPendingScheduledDraw;
1.1368 + if (Screen()->ChangeTracking())
1.1369 + {
1.1370 + iWsWin->AddDirtyWindowRegion(iCurrentSegment->iRegion); // stored in window coordinates
1.1371 + if (iWsWin->IsVisible())
1.1372 + {
1.1373 + // Window is visible, (we're ignoring whether it's obscured or not)
1.1374 + // we need to send the new draw commands to the render stage.
1.1375 + Screen()->ScheduleWindow(iWsWin);
1.1376 + }
1.1377 + }
1.1378 + if (!iWsWin->HasBeenDrawnToScreen())
1.1379 + {
1.1380 + CliWin()->ScheduleRegionUpdate(NULL);
1.1381 + }
1.1382 + else if(!Screen()->ChangeTracking() && (iWsWin->VisibleRegion().Count() || iWsWin->VisibleRegion().CheckError())) // on screen? good.
1.1383 + {
1.1384 + STACK_REGION draw; //### in low memory where VisibleRegion() is intact we can degrade much better than this!
1.1385 + draw.Copy(iCurrentSegment->iRegion);
1.1386 + draw.Offset(iWsWin->Origin());
1.1387 + draw.Intersect(iWsWin->VisibleRegion());
1.1388 + if(!draw.CheckError())
1.1389 + Screen()->AddRedrawRegion(draw);
1.1390 + else
1.1391 + Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
1.1392 + draw.Close();
1.1393 + }
1.1394 + }
1.1395 +
1.1396 + iCurrentSegment = NULL;
1.1397 +
1.1398 + //store the result of the current redraw
1.1399 + if(iFlags&ECurrentRedrawFailedStorage)
1.1400 + iFlags |= EPreviousRedrawFailedStorage; //set
1.1401 + else
1.1402 + iFlags &= ~EPreviousRedrawFailedStorage; //unset
1.1403 + iFlags &= ~ECurrentRedrawFailedStorage; //unset the flag for the next redraw
1.1404 + //
1.1405 +
1.1406 + iFlags&=~EBeginEndRedraw;
1.1407 + }
1.1408 +
1.1409 +void CWsRedrawMsgWindow::ValidateRect(const TRect *aRect)
1.1410 + {
1.1411 + if (!WsWin()->BaseParent())
1.1412 + OwnerPanic(EWservPanicParentDeleted);
1.1413 + if (aRect)
1.1414 + iRedrawRect = *aRect;
1.1415 + if (!iInvalid.IsEmpty())
1.1416 + {
1.1417 + STACK_REGION validated;
1.1418 + validated.Copy(iInvalid);
1.1419 + if (aRect)
1.1420 + validated.ClipRect(iRedrawRect);
1.1421 +
1.1422 + if (iInvalid.CheckError())
1.1423 + {
1.1424 + iInvalid.Copy(iWsWin->VisibleRegion());
1.1425 + iInvalid.Offset(-iWsWin->Origin());
1.1426 + }
1.1427 + iInvalid.SubRegion(validated);
1.1428 + validated.Close();
1.1429 + }
1.1430 + RemoveFromRedrawQueueIfEmpty();
1.1431 + }
1.1432 +
1.1433 +TRgb CWsRedrawMsgWindow::BackColor() const
1.1434 + {
1.1435 + return(iBackColor);
1.1436 + }
1.1437 +
1.1438 +/**
1.1439 +This function used to be quite clever about what it invalidated and what it redrew by copying
1.1440 +rectangles of the screen around. This is a lot less subtle, and makes calling Scroll pretty much
1.1441 +pointless, but it IS functionally correct.
1.1442 +*/
1.1443 +void CWsRedrawMsgWindow::Scroll(const TRect &aClipRect, const TPoint &aOffset,const TRect &aRect)
1.1444 + {
1.1445 + TRect rect = aRect;
1.1446 + rect.Intersection(aClipRect);
1.1447 + Invalidate(&rect);
1.1448 + rect = aRect;
1.1449 + rect.Move(aOffset);
1.1450 + rect.Intersection(aClipRect);
1.1451 + Invalidate(&rect);
1.1452 + }
1.1453 +
1.1454 +void CWsRedrawMsgWindow::ClearRedrawStore(TBool aClearPendingRedraw)
1.1455 + {
1.1456 + if(aClearPendingRedraw && (iFlags & EPendingScheduledDraw))
1.1457 + iFlags &= ~EPendingScheduledDraw;
1.1458 +
1.1459 + DiscardStoredCommands();
1.1460 + Invalidate();
1.1461 + }
1.1462 +
1.1463 +
1.1464 +void CWsRedrawMsgWindow::PrepareForResizeL(const TSize& aSize, TSize& /*aOldSize*/)
1.1465 + {
1.1466 + TBool anyIncreases(EFalse);
1.1467 + if (aSize.iWidth>iWsWin->Size().iWidth||aSize.iHeight>iWsWin->Size().iHeight)
1.1468 + {
1.1469 + anyIncreases = ETrue;
1.1470 + }
1.1471 +
1.1472 + TRect newWinRect(TPoint(0,0),aSize);
1.1473 + iInvalid.ClipRect(newWinRect);
1.1474 + if (anyIncreases)
1.1475 + {
1.1476 + // add new invalid region to iInvalid
1.1477 + iInvalid.AddRect(newWinRect);
1.1478 + QueueRedraw();
1.1479 + iWsWin->WsOwner()->TriggerRedraw();
1.1480 + }
1.1481 + }
1.1482 +
1.1483 +void CWsRedrawMsgWindow::Moved()
1.1484 + {
1.1485 + if (!(iFlags & EStoringEntireWindow))
1.1486 + {
1.1487 + DiscardSegmentsOutsideViewport();
1.1488 + }
1.1489 + if (iInvalid.Count())
1.1490 + {
1.1491 + QueueRedraw();
1.1492 + iWsWin->WsOwner()->TriggerRedraw();
1.1493 + }
1.1494 + }
1.1495 +
1.1496 +TBool CWsRedrawMsgWindow::Contains(const TArray<TGraphicDrawerId>& aDrawers,const TRegion& aRegion) const
1.1497 + {
1.1498 + if (iRedrawSegments.Count() > 0)
1.1499 + {
1.1500 + // scan redraw store: calls Contains() on every region drawing commands are stored for,
1.1501 + // looking for a DrawWsGraphic command that intersects the aRegion
1.1502 + TBool contains = EFalse;
1.1503 + const TInt regionCount = iRedrawSegments.Count();
1.1504 +
1.1505 + // Apply an origin correction. The input aRegion is screen-absolute, while the redraw regions are window-relative.
1.1506 + STACK_REGION relRegion;
1.1507 + relRegion.Copy(aRegion);
1.1508 + relRegion.Offset(-(CliWin()->Origin().iX), -(CliWin()->Origin().iY));
1.1509 +
1.1510 + // loop through regions, stops when a match is found
1.1511 + for (TInt regionNum = 0; (regionNum < regionCount) && !contains; ++regionNum)
1.1512 + {
1.1513 + contains = iRedrawSegments[regionNum]->ContainsDrawers(aDrawers, relRegion);
1.1514 + }
1.1515 + relRegion.Close();
1.1516 + return contains;
1.1517 + }
1.1518 + else
1.1519 + {
1.1520 + return CWsWindowRedraw::Contains(aDrawers,aRegion);
1.1521 + }
1.1522 + }
1.1523 +
1.1524 +TBool CWsRedrawMsgWindow::RedrawingInProgress() const
1.1525 + {
1.1526 + return (iFlags & EBeginEndRedraw);
1.1527 + }
1.1528 +
1.1529 +void CWsRedrawMsgWindow::WindowClosing()
1.1530 + {
1.1531 + iWsWin->WsOwner()->RedrawQueue()->RemoveInvalid(this);
1.1532 + CWsWindowRedraw::WindowClosing();
1.1533 + }
1.1534 +
1.1535 +TBool CWsRedrawMsgWindow::IsRedrawStoreEmpty() const
1.1536 + {
1.1537 + return (iRedrawSegments.Count() <= 0); // Begin and End redraw are in the store too.
1.1538 + }
1.1539 +
1.1540 +TBool CWsRedrawMsgWindow::IsBackgroundClearEnabled() const
1.1541 + {
1.1542 + return ((iFlags & EBackgroundClear) != 0);
1.1543 + }
1.1544 +
1.1545 +void CWsRedrawMsgWindow::CleanupBitmapRefArray(const RArray<TInt>& aHandleArray)
1.1546 + {
1.1547 + TInt count = aHandleArray.Count();
1.1548 + for(TInt ii = count - 1 ; ii >= 0; ii--)
1.1549 + {
1.1550 + const TInt indexBitmapRef = iFbsBitmapRefArray.FindInOrder(aHandleArray[ii], &FindBitmapRefByHandle);
1.1551 + WS_ASSERT_DEBUG(indexBitmapRef >= 0,EWsPanicUnexpectedBitmapHandleInArray);
1.1552 + if (indexBitmapRef>=0)
1.1553 + {
1.1554 + iFbsBitmapRefArray[indexBitmapRef]->DecRefCount();
1.1555 + if (iFbsBitmapRefArray[indexBitmapRef]->RefCount()==0)
1.1556 + {
1.1557 + delete iFbsBitmapRefArray[indexBitmapRef];
1.1558 + iFbsBitmapRefArray.Remove(indexBitmapRef);
1.1559 + }
1.1560 + }
1.1561 + }
1.1562 + }
1.1563 +
1.1564 +void CWsRedrawMsgWindow::SetScope(TScope aScope)
1.1565 + {
1.1566 + if (aScope == EStoreEntireWindow)
1.1567 + {
1.1568 + if (!(iFlags & EStoringEntireWindow))
1.1569 + {
1.1570 + iFlags |= EStoringEntireWindow;
1.1571 + Invalidate();
1.1572 + }
1.1573 + }
1.1574 + else
1.1575 + {
1.1576 + if (iFlags & EStoringEntireWindow)
1.1577 + {
1.1578 + iFlags &= ~ EStoringEntireWindow;
1.1579 + DiscardSegmentsOutsideViewport();
1.1580 + }
1.1581 + }
1.1582 + }
1.1583 +
1.1584 +/**
1.1585 +Removes all segments from the redraw store which are outside the viewport onto the window.
1.1586 +Note that this doesn't clip the regions of those segments which are partly outside, since
1.1587 +this wouldn't actually achieve anything useful.
1.1588 +
1.1589 +This function allocates memory so it is not suitable to run as part of ReleaseMemory.
1.1590 +*/
1.1591 +TBool CWsRedrawMsgWindow::DiscardSegmentsOutsideViewport()
1.1592 + {
1.1593 + TBool discarded = EFalse;
1.1594 + TInt count = iRedrawSegments.Count();
1.1595 + STACK_REGION viewport;
1.1596 + CliWin()->GetClippedBaseArea(viewport);
1.1597 + viewport.Offset(-iWsWin->Origin());
1.1598 + STACK_REGION intersect;
1.1599 + for (TInt idx = count - 1; idx >= 0; --idx)
1.1600 + {
1.1601 + CRedrawSegment * segment = iRedrawSegments[idx];
1.1602 + intersect.Intersection(segment->iRegion, viewport);
1.1603 + if (!intersect.CheckError() && intersect.IsEmpty())
1.1604 + {
1.1605 + iInvalid.Union(segment->iRegion);
1.1606 + delete segment;
1.1607 + iRedrawSegments.Remove(idx);
1.1608 + if (iCurrentSegment == segment)
1.1609 + iCurrentSegment = NULL;
1.1610 + discarded = ETrue;
1.1611 + }
1.1612 + }
1.1613 + intersect.Close();
1.1614 + viewport.Close();
1.1615 + return discarded;
1.1616 + }
1.1617 +
1.1618 +/**
1.1619 +Statements encapsulated in between Lock() and Unlock() is guaranteed to execute in an
1.1620 +atomic way without being interupted by a call to ReleaseMemory from CWsMemoryManager.
1.1621 +Locking will prevent memory belonging to this object to be freed during a
1.1622 +memory alloc/realloc originating from self.
1.1623 +*/
1.1624 +void CWsRedrawMsgWindow::Lock()
1.1625 + {
1.1626 + ++iMemoryLock;
1.1627 + }
1.1628 +
1.1629 +void CWsRedrawMsgWindow::Unlock()
1.1630 + {
1.1631 + --iMemoryLock;
1.1632 + WS_ASSERT_DEBUG(iMemoryLock >= 0, EWsPanicMemoryLock);
1.1633 + }
1.1634 +
1.1635 +TBool CWsRedrawMsgWindow::ReleaseMemory(MWsMemoryRelease::TMemoryReleaseLevel aLevel)
1.1636 + {
1.1637 + //When this function is called, wserv is in the middle of executing something.
1.1638 + //Therefore we can not safely do anything that alters the state of any shared
1.1639 + //resouces (like e.g. CScreenRedraw::iInvalid).
1.1640 + //In addition, we should refrain from anything that might try to allocate memory.
1.1641 + TBool released = EFalse;
1.1642 + //Don't release iRedrawSegments from this win if its currently being rendered,
1.1643 + //is releasing memory or is receiving drawcommands.
1.1644 + if (iMemoryLock == 0 && !iCurrentSegment)
1.1645 + {
1.1646 + Lock();
1.1647 + switch (aLevel)
1.1648 + {
1.1649 + case MWsMemoryRelease::ELow:
1.1650 + break;
1.1651 + case MWsMemoryRelease::EMedium:
1.1652 + break;
1.1653 + case MWsMemoryRelease::EHigh:
1.1654 + //Only release memory from background windows.
1.1655 + if (iRedrawSegments.Count() > 0 && iWsWin->VisibleRegion().IsEmpty())
1.1656 + {
1.1657 + ReleaseRedrawSegments();
1.1658 + released = ETrue;
1.1659 + }
1.1660 + break;
1.1661 + }
1.1662 + Unlock();
1.1663 + }
1.1664 + return released;
1.1665 + }
1.1666 +
1.1667 +void CWsRedrawMsgWindow::ReleaseRedrawSegments()
1.1668 + {
1.1669 + iLastDrawGc = NULL;
1.1670 + iCurrentSegment = NULL;
1.1671 + iRedrawSegments.ResetAndDestroy();
1.1672 +
1.1673 + //The call to ResetAndDestroy just freed some memory so it should be
1.1674 + //possible to call Invalidate() now.
1.1675 + Invalidate();
1.1676 +
1.1677 + //Releasing the same window over and over again could quickly end up in
1.1678 + //a never ending loop with a high-prio client before we find the window
1.1679 + //that has nicked all memory. So call accessed now to prevent that.
1.1680 + iWsWin->Accessed();
1.1681 + }
1.1682 +
1.1683 +void CWsRedrawMsgWindow::VisibleRegionChange()
1.1684 + {
1.1685 + if (!iFlags & EStoringEntireWindow)
1.1686 + {
1.1687 + DiscardSegmentsOutsideViewport();
1.1688 + }
1.1689 + if ((!iInvalid.IsEmpty()) && (!iWsWin->VisibleRegion().IsEmpty()))
1.1690 + {
1.1691 + STACK_REGION exposed;
1.1692 + exposed.Copy(iInvalid);
1.1693 + exposed.Offset(iWsWin->Origin());
1.1694 + exposed.Intersect(iWsWin->VisibleRegion());
1.1695 + if (!exposed.IsEmpty())
1.1696 + {
1.1697 + QueueRedraw();
1.1698 + }
1.1699 + exposed.Close();
1.1700 + }
1.1701 + }
1.1702 +
1.1703 +TBool CWsRedrawMsgWindow::ReadyToDraw() const
1.1704 + {
1.1705 + //We are only ready to draw when we have a complete segment.
1.1706 + if (iWsWin->HasBeenDrawnToScreen())
1.1707 + return ETrue;
1.1708 +
1.1709 + if (iRedrawSegments.Count() == 0)
1.1710 + return EFalse;
1.1711 +
1.1712 + if (iRedrawSegments.Count() > 1)
1.1713 + return ETrue;
1.1714 +
1.1715 + if (iRedrawSegments[0]->iRedrawSegmentType == ESegmentTypePendingRedraw)
1.1716 + return EFalse;
1.1717 +
1.1718 + return ETrue;
1.1719 + }
1.1720 +
1.1721 +TInt CWsRedrawMsgWindow::SizeInBytes() const
1.1722 + {
1.1723 + TInt size = sizeof(CWsRedrawMsgWindow);
1.1724 + for(TInt i = iRedrawSegments.Count()-1; i >= 0; i--)
1.1725 + {
1.1726 + size += iRedrawSegments[i]->SizeInBytes();
1.1727 + }
1.1728 + size += iInvalid.Count() * sizeof(TRect);
1.1729 + size += iLocalRedrawRegion.Count() * sizeof(TRect);
1.1730 + return size;
1.1731 + }
1.1732 +
1.1733 +void CWsRedrawMsgWindow::PromotePendingSegment()
1.1734 + {
1.1735 + if (iRedrawSegments.Count() > 0 && iRedrawSegments[iRedrawSegments.Count() - 1]->iRedrawSegmentType == ESegmentTypePendingRedraw)
1.1736 + {
1.1737 + CRedrawSegment * segment = iRedrawSegments[iRedrawSegments.Count() - 1];
1.1738 + const TRect * rect = segment->iRegion.RectangleList();
1.1739 + // when we get here there should only ever be one rectangle in the region, but we are playing safe
1.1740 + for (TInt r = 0; r < segment->iRegion.Count(); ++r)
1.1741 + {
1.1742 + SubtractRectFromSegmentArray(*rect);
1.1743 + ++rect;
1.1744 + }
1.1745 + segment->iRedrawSegmentType = ESegmentTypeRedraw;
1.1746 + }
1.1747 + }
1.1748 +
1.1749 +CFbsBitmapRef::CFbsBitmapRef()
1.1750 + {
1.1751 + }
1.1752 +
1.1753 +CFbsBitmapRef::~CFbsBitmapRef()
1.1754 + {
1.1755 + WS_ASSERT_DEBUG(iRefCount==0,EWsPanicCounterValue);
1.1756 + }