First public contribution.
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 // This class deals with drawing from backup bitmap
19 #include "backedupwindow.h"
31 #include "playbackgc.h"
32 #include "devicemap.h"
34 CFbsBitGc *CWsBackedUpWindow::iBitGc=NULL;
36 CWsBackedUpWindow::CWsBackedUpWindow(CWsWindow *aWin, TDisplayMode aDisplayMode)
37 : CWsWindowRedraw(aWin), iDisplayMode(aDisplayMode)
40 void CWsBackedUpWindow::StaticInitL()
42 iBitGc=CFbsBitGc::NewL();
45 void CWsBackedUpWindow::StaticDestroy()
51 void CWsBackedUpWindow::ActivateGc()
53 iBitGc->Activate(iBitmapDevice);
55 iBitGc->SetBrushColor(BackColor());
58 TBool CWsBackedUpWindow::DrawCommand(CWsGc*,const TAny*)
60 if (Screen()->ChangeTracking())
61 MarkDirtyAndSchedule(iCurrentRegion);
63 Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
68 void CWsBackedUpWindow::ConstructL()
70 iDisplayMode=iWsWin->SetRequiredDisplayModeL(iDisplayMode);
71 TSize size=iWsWin->Size();
72 iBitmap=new(ELeave) CFbsBitmap();
73 User::LeaveIfError(iBitmap->Create(size, iDisplayMode));
74 iBitmapDevice=CFbsBitmapDevice::NewL(iBitmap);
78 iBitGc->SetDrawMode(CGraphicsContext::EDrawModeWriteAlpha);
79 iBitGc->Clear(TRect(size));
80 iBitGc->SetDrawMode(CGraphicsContext::EDrawModePEN);
81 WS_ASSERT_DEBUG(iWsWin->WinType()==EWinTypeClient,EWsPanicWindowType);
84 void CWsBackedUpWindow::PrepareForResizeL(const TSize &aSize, TSize &aOldSize)
86 aOldSize=iBitmapDevice->SizeInPixels();
89 User::LeaveIfError(iBitmapDevice->Resize(aSize));
94 void CWsBackedUpWindow::Resize(const TSize &aSize, const TSize &aOldSize)
97 iBitGc->SetClippingRegion(NULL);
98 iBitGc->Clear(TRect(aOldSize.iWidth, 0, aSize.iWidth, aOldSize.iHeight));
99 iBitGc->Clear(TRect(0, aOldSize.iHeight,aSize.iWidth, aSize.iHeight));
101 static_cast<CWsClientWindow *>(iWsWin)->ReactivateGcs();
103 if(Screen()->ChangeTracking())
105 //Keep track of the region we need to refresh when we recieve draw commands
106 iCurrentRegion.Reset();
107 iCurrentRegion.Copy(iWsWin->WindowArea());
108 iCurrentRegion.Offset(-iWsWin->Origin());
109 iCurrentRegion.Tidy();
110 if(iCurrentRegion.CheckError())
112 iCurrentRegion.Reset();
113 TRegionFix<1> fallback(iWsWin->AbsRect());
114 iCurrentRegion.Copy(fallback);
115 iCurrentRegion.Offset(-iWsWin->Origin());
118 //If the window has nerver been drawn to screen, we now schedule the initial draw. This can't
119 //be done in ConstructL because BackedUpWindows are created with size(0,0). And we can't check
120 //iWsWin->IsActive() because the client might activate the window before giving it a size.
121 if (!iHasBeenScheduled)
123 iHasBeenScheduled = ETrue;
124 MarkDirtyAndSchedule(iCurrentRegion);
129 CWsBackedUpWindow::~CWsBackedUpWindow()
131 iCurrentRegion.Reset();
132 delete iBitmapDevice;
136 TBool CWsBackedUpWindow::CommandL(TInt aOpcode, TWsWinCmdUnion &aCmd)
140 case EWsWinOpUpdateBackupBitmap:
142 case EWsWinOpMaintainBackup:
144 case EWsWinOpBitmapHandle:
145 SetReply(iBitmap->Handle());
147 case EWsWinOpUpdateScreen:
149 TRegionFix<1> fixRegion(iWsWin->AbsRect());
150 UpdateScreen(fixRegion);
153 case EWsWinOpUpdateScreenRegion:
155 RWsRegion *clientRegion=NULL;
156 TRAPD(err,clientRegion=GetRegionFromClientL(iWsWin->WsOwner(), *aCmd.Int));
157 if (err==KErrNone && !clientRegion->CheckError())
159 clientRegion->Offset(iWsWin->Origin());
160 clientRegion->ClipRect(iWsWin->AbsRect());
161 UpdateScreen(*clientRegion);
165 TRegionFix<1> fixRegion(iWsWin->AbsRect());
166 UpdateScreen(fixRegion);
168 clientRegion->Destroy();
171 case EWsWinOpSetBackgroundSurface:
172 case EWsWinOpSetBackgroundSurfaceConfig:
173 case EWsWinOpGetBackgroundSurfaceConfig:
174 OwnerPanic(EWservPanicDrawable); // Backed up windows don't support these
182 CWsBackedUpWindow *CWsBackedUpWindow::Backup() const
184 return((CWsBackedUpWindow *)this);
187 CFbsDevice* CWsBackedUpWindow::OutputDevice() const
189 return iBitmapDevice;
192 TRgb CWsBackedUpWindow::BackColor() const
194 return(iWsWin->RootWindow()->DefaultBackgroundColor());
197 void CWsBackedUpWindow::Scroll(const TRect &aClipRect, const TPoint &aOffset,const TRect &aRect)
199 TRect winBorder=TRect(iWsWin->Size());
200 TRect clipRect=aClipRect;
201 TRect srcRect = aRect;
202 clipRect.Intersection(winBorder);
203 if (!clipRect.IsEmpty())
204 { // If we have to do something (a visible part will change)
205 srcRect.Intersection(clipRect);
207 STACK_REGION regionToClear;
208 regionToClear.AddRect(aRect);
209 regionToClear.SubRect(srcRect);
210 regionToClear.Offset(aOffset);
213 iBitGc->SetClippingRect(clipRect);
214 iBitGc->SetDrawMode(CGraphicsContext::EDrawModeWriteAlpha);
215 iBitGc->CopyRect(aOffset,srcRect);
216 for (TInt k=0;k<regionToClear.Count();k++)
218 iBitGc->Clear(regionToClear[k]);
220 iBitGc->SetClippingRect(winBorder);
221 iBitGc->SetDrawMode(CGraphicsContext::EDrawModePEN);
222 TRegionFix<1> fixRegion(iWsWin->AbsRect());
223 UpdateScreen(fixRegion);
224 regionToClear.Close();
228 TBool CWsBackedUpWindow::NeedsRedraw() const
233 TBool CWsBackedUpWindow::GetRedrawRect(TRect &) const
238 void CWsBackedUpWindow::SetSizeInTwips()
240 TSize size=iBitmap->SizeInPixels();
241 size.iWidth=Screen()->DeviceMap().HorizontalPixelsToTwips(size.iWidth);
242 size.iHeight=Screen()->DeviceMap().VerticalPixelsToTwips(size.iHeight);
243 iBitmap->SetSizeInTwips(size);
246 void CWsBackedUpWindow::DrawWindow()
248 MWsGraphicsContext* gc = static_cast<MWsGraphicsContext*>(CPlaybackGc::Instance()->ResolveObjectInterface(KMWsGraphicsContext));
250 gc->SetOrigin(iWsWin->Origin());
251 gc->SetClippingRegion(*iRedrawRegion);
252 gc->BitBlt(TPoint(0,0), *iBitmap);
256 This function updates the window's dirty region and schedules a redraw if needed.
257 Only used when the screen is run in CHANGETRACKING mode.
258 @param aRegion in window coordinates
260 void CWsBackedUpWindow::MarkDirtyAndSchedule(const TRegion& aRegion)
262 WS_ASSERT_DEBUG(Screen()->ChangeTracking(),EWsPanicNoChangetracking);
264 if(!aRegion.IsEmpty())
266 iWsWin->AddDirtyWindowRegion(aRegion);
267 if (iWsWin->IsActive() && iWsWin->IsVisible())
269 Screen()->ScheduleWindow(iWsWin);
275 This function selects the desired behaviour depending on whether the screen is
276 run in CHANGETRACKING mode or not.
277 @param aRegion in screen coordinates
279 void CWsBackedUpWindow::UpdateScreen(const TRegion& aRegion)
281 if (Screen()->ChangeTracking())
284 region.Copy(aRegion);
285 region.Offset(-WsWin()->Origin()); //convert to window coordinates
286 MarkDirtyAndSchedule(region);
291 Screen()->AddRedrawRegion(aRegion);