1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/windowing/windowserver/nga/SERVER/openwfc/backedupwindow.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,293 @@
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 +// This class deals with drawing from backup bitmap
1.19 +//
1.20 +//
1.21 +
1.22 +#include "backedupwindow.h"
1.23 +#include "server.h"
1.24 +#include "gc.h"
1.25 +#include "wstop.h"
1.26 +#include "ANIM.H"
1.27 +#include "EVQUEUE.H"
1.28 +#include <s32mem.h>
1.29 +#include <gdi.h>
1.30 +#include "panics.h"
1.31 +#include "inifile.h"
1.32 +#include "rootwin.h"
1.33 +#include "EVENT.H"
1.34 +#include "playbackgc.h"
1.35 +#include "devicemap.h"
1.36 +
1.37 +CFbsBitGc *CWsBackedUpWindow::iBitGc=NULL;
1.38 +
1.39 +CWsBackedUpWindow::CWsBackedUpWindow(CWsWindow *aWin, TDisplayMode aDisplayMode)
1.40 + : CWsWindowRedraw(aWin), iDisplayMode(aDisplayMode)
1.41 + {}
1.42 +
1.43 +void CWsBackedUpWindow::StaticInitL()
1.44 + {
1.45 + iBitGc=CFbsBitGc::NewL();
1.46 + }
1.47 +
1.48 +void CWsBackedUpWindow::StaticDestroy()
1.49 + {
1.50 + delete iBitGc;
1.51 + iBitGc = 0;
1.52 + }
1.53 +
1.54 +void CWsBackedUpWindow::ActivateGc()
1.55 + {
1.56 + iBitGc->Activate(iBitmapDevice);
1.57 + iBitGc->Reset();
1.58 + iBitGc->SetBrushColor(BackColor());
1.59 + }
1.60 +
1.61 +TBool CWsBackedUpWindow::DrawCommand(CWsGc*,const TAny*)
1.62 + {
1.63 + if (Screen()->ChangeTracking())
1.64 + MarkDirtyAndSchedule(iCurrentRegion);
1.65 + else
1.66 + Screen()->AddRedrawRegion(iWsWin->VisibleRegion());
1.67 +
1.68 + return ETrue;
1.69 + }
1.70 +
1.71 +void CWsBackedUpWindow::ConstructL()
1.72 + {
1.73 + iDisplayMode=iWsWin->SetRequiredDisplayModeL(iDisplayMode);
1.74 + TSize size=iWsWin->Size();
1.75 + iBitmap=new(ELeave) CFbsBitmap();
1.76 + User::LeaveIfError(iBitmap->Create(size, iDisplayMode));
1.77 + iBitmapDevice=CFbsBitmapDevice::NewL(iBitmap);
1.78 + SetSizeInTwips();
1.79 +//
1.80 + ActivateGc();
1.81 + iBitGc->SetDrawMode(CGraphicsContext::EDrawModeWriteAlpha);
1.82 + iBitGc->Clear(TRect(size));
1.83 + iBitGc->SetDrawMode(CGraphicsContext::EDrawModePEN);
1.84 + WS_ASSERT_DEBUG(iWsWin->WinType()==EWinTypeClient,EWsPanicWindowType);
1.85 + }
1.86 +
1.87 +void CWsBackedUpWindow::PrepareForResizeL(const TSize &aSize, TSize &aOldSize)
1.88 + {
1.89 + aOldSize=iBitmapDevice->SizeInPixels();
1.90 + if (aOldSize!=aSize)
1.91 + {
1.92 + User::LeaveIfError(iBitmapDevice->Resize(aSize));
1.93 + SetSizeInTwips();
1.94 + }
1.95 + }
1.96 +
1.97 +void CWsBackedUpWindow::Resize(const TSize &aSize, const TSize &aOldSize)
1.98 + {
1.99 + ActivateGc();
1.100 + iBitGc->SetClippingRegion(NULL);
1.101 + iBitGc->Clear(TRect(aOldSize.iWidth, 0, aSize.iWidth, aOldSize.iHeight));
1.102 + iBitGc->Clear(TRect(0, aOldSize.iHeight,aSize.iWidth, aSize.iHeight));
1.103 +
1.104 + static_cast<CWsClientWindow *>(iWsWin)->ReactivateGcs();
1.105 +
1.106 + if(Screen()->ChangeTracking())
1.107 + {
1.108 + //Keep track of the region we need to refresh when we recieve draw commands
1.109 + iCurrentRegion.Reset();
1.110 + iCurrentRegion.Copy(iWsWin->WindowArea());
1.111 + iCurrentRegion.Offset(-iWsWin->Origin());
1.112 + iCurrentRegion.Tidy();
1.113 + if(iCurrentRegion.CheckError())
1.114 + {
1.115 + iCurrentRegion.Reset();
1.116 + TRegionFix<1> fallback(iWsWin->AbsRect());
1.117 + iCurrentRegion.Copy(fallback);
1.118 + iCurrentRegion.Offset(-iWsWin->Origin());
1.119 + }
1.120 +
1.121 + //If the window has nerver been drawn to screen, we now schedule the initial draw. This can't
1.122 + //be done in ConstructL because BackedUpWindows are created with size(0,0). And we can't check
1.123 + //iWsWin->IsActive() because the client might activate the window before giving it a size.
1.124 + if (!iHasBeenScheduled)
1.125 + {
1.126 + iHasBeenScheduled = ETrue;
1.127 + MarkDirtyAndSchedule(iCurrentRegion);
1.128 + }
1.129 + }
1.130 + }
1.131 +
1.132 +CWsBackedUpWindow::~CWsBackedUpWindow()
1.133 + {
1.134 + iCurrentRegion.Reset();
1.135 + delete iBitmapDevice;
1.136 + delete iBitmap;
1.137 + }
1.138 +
1.139 +TBool CWsBackedUpWindow::CommandL(TInt aOpcode, TWsWinCmdUnion &aCmd)
1.140 + {
1.141 + switch(aOpcode)
1.142 + {
1.143 + case EWsWinOpUpdateBackupBitmap:
1.144 + break;
1.145 + case EWsWinOpMaintainBackup:
1.146 + break;
1.147 + case EWsWinOpBitmapHandle:
1.148 + SetReply(iBitmap->Handle());
1.149 + break;
1.150 + case EWsWinOpUpdateScreen:
1.151 + {
1.152 + TRegionFix<1> fixRegion(iWsWin->AbsRect());
1.153 + UpdateScreen(fixRegion);
1.154 + }
1.155 + break;
1.156 + case EWsWinOpUpdateScreenRegion:
1.157 + {
1.158 + RWsRegion *clientRegion=NULL;
1.159 + TRAPD(err,clientRegion=GetRegionFromClientL(iWsWin->WsOwner(), *aCmd.Int));
1.160 + if (err==KErrNone && !clientRegion->CheckError())
1.161 + {
1.162 + clientRegion->Offset(iWsWin->Origin());
1.163 + clientRegion->ClipRect(iWsWin->AbsRect());
1.164 + UpdateScreen(*clientRegion);
1.165 + }
1.166 + else
1.167 + {
1.168 + TRegionFix<1> fixRegion(iWsWin->AbsRect());
1.169 + UpdateScreen(fixRegion);
1.170 + }
1.171 + clientRegion->Destroy();
1.172 + }
1.173 + break;
1.174 + case EWsWinOpSetBackgroundSurface:
1.175 + case EWsWinOpSetBackgroundSurfaceConfig:
1.176 + case EWsWinOpGetBackgroundSurfaceConfig:
1.177 + OwnerPanic(EWservPanicDrawable); // Backed up windows don't support these
1.178 + break;
1.179 + default:
1.180 + return(EFalse);
1.181 + }
1.182 + return(ETrue);
1.183 + }
1.184 +
1.185 +CWsBackedUpWindow *CWsBackedUpWindow::Backup() const
1.186 + {
1.187 + return((CWsBackedUpWindow *)this);
1.188 + }
1.189 +
1.190 +CFbsDevice* CWsBackedUpWindow::OutputDevice() const
1.191 + {
1.192 + return iBitmapDevice;
1.193 + }
1.194 +
1.195 +TRgb CWsBackedUpWindow::BackColor() const
1.196 + {
1.197 + return(iWsWin->RootWindow()->DefaultBackgroundColor());
1.198 + }
1.199 +
1.200 +void CWsBackedUpWindow::Scroll(const TRect &aClipRect, const TPoint &aOffset,const TRect &aRect)
1.201 + {
1.202 + TRect winBorder=TRect(iWsWin->Size());
1.203 + TRect clipRect=aClipRect;
1.204 + TRect srcRect = aRect;
1.205 + clipRect.Intersection(winBorder);
1.206 + if (!clipRect.IsEmpty())
1.207 + { // If we have to do something (a visible part will change)
1.208 + srcRect.Intersection(clipRect);
1.209 +
1.210 + STACK_REGION regionToClear;
1.211 + regionToClear.AddRect(aRect);
1.212 + regionToClear.SubRect(srcRect);
1.213 + regionToClear.Offset(aOffset);
1.214 +
1.215 + ActivateGc();
1.216 + iBitGc->SetClippingRect(clipRect);
1.217 + iBitGc->SetDrawMode(CGraphicsContext::EDrawModeWriteAlpha);
1.218 + iBitGc->CopyRect(aOffset,srcRect);
1.219 + for (TInt k=0;k<regionToClear.Count();k++)
1.220 + {
1.221 + iBitGc->Clear(regionToClear[k]);
1.222 + }
1.223 + iBitGc->SetClippingRect(winBorder);
1.224 + iBitGc->SetDrawMode(CGraphicsContext::EDrawModePEN);
1.225 + TRegionFix<1> fixRegion(iWsWin->AbsRect());
1.226 + UpdateScreen(fixRegion);
1.227 + regionToClear.Close();
1.228 + }
1.229 + }
1.230 +
1.231 +TBool CWsBackedUpWindow::NeedsRedraw() const
1.232 + {
1.233 + return(EFalse);
1.234 + }
1.235 +
1.236 +TBool CWsBackedUpWindow::GetRedrawRect(TRect &) const
1.237 + {
1.238 + return(EFalse);
1.239 + }
1.240 +
1.241 +void CWsBackedUpWindow::SetSizeInTwips()
1.242 + {
1.243 + TSize size=iBitmap->SizeInPixels();
1.244 + size.iWidth=Screen()->DeviceMap().HorizontalPixelsToTwips(size.iWidth);
1.245 + size.iHeight=Screen()->DeviceMap().VerticalPixelsToTwips(size.iHeight);
1.246 + iBitmap->SetSizeInTwips(size);
1.247 + }
1.248 +
1.249 +void CWsBackedUpWindow::DrawWindow()
1.250 + {
1.251 + MWsGraphicsContext* gc = static_cast<MWsGraphicsContext*>(CPlaybackGc::Instance()->ResolveObjectInterface(KMWsGraphicsContext));
1.252 +
1.253 + gc->SetOrigin(iWsWin->Origin());
1.254 + gc->SetClippingRegion(*iRedrawRegion);
1.255 + gc->BitBlt(TPoint(0,0), *iBitmap);
1.256 + }
1.257 +
1.258 +/**
1.259 +This function updates the window's dirty region and schedules a redraw if needed.
1.260 +Only used when the screen is run in CHANGETRACKING mode.
1.261 +@param aRegion in window coordinates
1.262 +*/
1.263 +void CWsBackedUpWindow::MarkDirtyAndSchedule(const TRegion& aRegion)
1.264 + {
1.265 + WS_ASSERT_DEBUG(Screen()->ChangeTracking(),EWsPanicNoChangetracking);
1.266 +
1.267 + if(!aRegion.IsEmpty())
1.268 + {
1.269 + iWsWin->AddDirtyWindowRegion(aRegion);
1.270 + if (iWsWin->IsActive() && iWsWin->IsVisible())
1.271 + {
1.272 + Screen()->ScheduleWindow(iWsWin);
1.273 + }
1.274 + }
1.275 + }
1.276 +
1.277 +/**
1.278 +This function selects the desired behaviour depending on whether the screen is
1.279 +run in CHANGETRACKING mode or not.
1.280 +@param aRegion in screen coordinates
1.281 +*/
1.282 +void CWsBackedUpWindow::UpdateScreen(const TRegion& aRegion)
1.283 + {
1.284 + if (Screen()->ChangeTracking())
1.285 + {
1.286 + STACK_REGION region;
1.287 + region.Copy(aRegion);
1.288 + region.Offset(-WsWin()->Origin()); //convert to window coordinates
1.289 + MarkDirtyAndSchedule(region);
1.290 + region.Close();
1.291 + }
1.292 + else
1.293 + {
1.294 + Screen()->AddRedrawRegion(aRegion);
1.295 + }
1.296 + }