os/graphics/windowing/windowserver/nga/SERVER/openwfc/screen.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2006-2010 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#include "screen.h"
sl@0
    17
sl@0
    18
#include <hal.h>
sl@0
    19
#include <graphics/wsscreendevice.h>
sl@0
    20
#include <graphics/wsscene.h>
sl@0
    21
#include <graphics/wselement.h>
sl@0
    22
#include "server.h"
sl@0
    23
#include "wstop.h"
sl@0
    24
#include "rootwin.h"
sl@0
    25
#include "walkwindowtree.h"
sl@0
    26
#include "EVENT.H"
sl@0
    27
#include "windowgroup.h"
sl@0
    28
#include "inifile.h"
sl@0
    29
#include "pointer.h"
sl@0
    30
#include "windowelementset.h"
sl@0
    31
#include "registeredsurfacemap.h"
sl@0
    32
#include "debugbar.h"
sl@0
    33
#include "ScreenRedraw.h"
sl@0
    34
#include "wspluginmanager.h"
sl@0
    35
#include "devicemap.h"
sl@0
    36
#include <graphics/wsdisplaymapping.h>
sl@0
    37
#if defined(__WINS__) && defined(_DEBUG)
sl@0
    38
#include "../../debuglog/osbwin.h"
sl@0
    39
#endif
sl@0
    40
#include <graphics/wsdisplaycontrol.h>
sl@0
    41
sl@0
    42
GLREF_D CDebugLogBase *wsDebugLog;
sl@0
    43
sl@0
    44
LOCAL_C inline MWsScene::TSceneRotation GcToScreen(CFbsBitGc::TGraphicsOrientation aGcOrientation)
sl@0
    45
	{
sl@0
    46
	MWsScene::TSceneRotation screenRotation = MWsScene::ESceneAntiClockwise0;
sl@0
    47
sl@0
    48
	switch (aGcOrientation)
sl@0
    49
		{
sl@0
    50
		case CFbsBitGc::EGraphicsOrientationRotated90:
sl@0
    51
			screenRotation = MWsScene::ESceneAntiClockwise90;
sl@0
    52
			break;
sl@0
    53
		case CFbsBitGc::EGraphicsOrientationRotated180:
sl@0
    54
			screenRotation = MWsScene::ESceneAntiClockwise180;
sl@0
    55
			break;
sl@0
    56
		case CFbsBitGc::EGraphicsOrientationRotated270:
sl@0
    57
			screenRotation = MWsScene::ESceneAntiClockwise270;
sl@0
    58
			break;
sl@0
    59
		}
sl@0
    60
sl@0
    61
	return screenRotation;
sl@0
    62
	}
sl@0
    63
sl@0
    64
LOCAL_C inline TDeviceOrientation GcToDevice(CFbsBitGc::TGraphicsOrientation aGcOrientation)
sl@0
    65
	{
sl@0
    66
	// Each device orientation is defined to be the value where just the bit
sl@0
    67
	// corresponding to the GDI orientation value is set.
sl@0
    68
	return (TDeviceOrientation)(1 << aGcOrientation);
sl@0
    69
	}
sl@0
    70
sl@0
    71
LOCAL_D TBool FindNextValue(TLex& aLex, TInt& aValue) // assumes the list cannot contain *negative* integers
sl@0
    72
	{
sl@0
    73
	while (!aLex.Eos())
sl@0
    74
		{
sl@0
    75
		const TUint character=aLex.Peek();
sl@0
    76
		if (Rng(TUint('0'), character, TUint('9')) || (character=='-'))
sl@0
    77
			{
sl@0
    78
			break;
sl@0
    79
			}
sl@0
    80
		aLex.Inc();
sl@0
    81
		}
sl@0
    82
sl@0
    83
	return (aLex.Val(aValue)==KErrNone);
sl@0
    84
	}
sl@0
    85
sl@0
    86
CScreen::CFallbackMap * CScreen::CFallbackMap::NewL(CScreen* aScreen)
sl@0
    87
	{
sl@0
    88
	CFallbackMap * self = new (ELeave) CFallbackMap(aScreen);
sl@0
    89
	CleanupStack::PushL(self);
sl@0
    90
	self->ConstructL();
sl@0
    91
	CleanupStack::Pop(self);
sl@0
    92
	return self;
sl@0
    93
	}
sl@0
    94
	
sl@0
    95
CScreen::CFallbackMap::CFallbackMap(CScreen* aScreen) :
sl@0
    96
	iScreen(aScreen),
sl@0
    97
	iRegion(TRect(TPoint(0,0), TSize(1,1))),
sl@0
    98
	iCount(0)
sl@0
    99
	{
sl@0
   100
	}
sl@0
   101
	
sl@0
   102
CScreen::CFallbackMap::~CFallbackMap()
sl@0
   103
	{
sl@0
   104
	delete []iMap;
sl@0
   105
	}
sl@0
   106
	
sl@0
   107
void CScreen::CFallbackMap::ConstructL()
sl@0
   108
	{
sl@0
   109
	iMapSize = 0;
sl@0
   110
	
sl@0
   111
	for (TInt num = 0; num < iScreen->NumScreenSizeModes(); ++num)
sl@0
   112
		{
sl@0
   113
		if (iScreen->IsValidScreenSizeMode(num))
sl@0
   114
			{
sl@0
   115
			const TSizeMode & mode = iScreen->ScreenSizeModeData(num);
sl@0
   116
			TInt width = mode.iScreenSize.iWidth / 32;
sl@0
   117
			if (mode.iScreenSize.iWidth & (32 - 1))
sl@0
   118
				++width;
sl@0
   119
			TInt size = width * mode.iScreenSize.iHeight;
sl@0
   120
			if (size > iMapSize)
sl@0
   121
				iMapSize = size;
sl@0
   122
			}
sl@0
   123
		}
sl@0
   124
		
sl@0
   125
	iMap = new (ELeave) TInt [iMapSize];
sl@0
   126
	}
sl@0
   127
sl@0
   128
void CScreen::CFallbackMap::Prepare()
sl@0
   129
	{
sl@0
   130
	const TSizeMode & mode = iScreen->ScreenSizeModeData();
sl@0
   131
	Mem::FillZ(iMap, iMapSize * sizeof(TInt));
sl@0
   132
	iCount = mode.iScreenSize.iHeight * mode.iScreenSize.iWidth;
sl@0
   133
	WS_ASSERT_DEBUG(iRegion.Count() == 1, EWsPanicScreenFallback);
sl@0
   134
	iRegion.Clear();
sl@0
   135
	iRegion.AddRect(TRect(mode.iOrigin, mode.iScreenSize));
sl@0
   136
	}
sl@0
   137
sl@0
   138
TBool CScreen::CFallbackMap::FillRegion(const TRegion& aRegion)
sl@0
   139
	{
sl@0
   140
	WS_ASSERT_DEBUG(!aRegion.CheckError(), EWsPanicScreenFallback);
sl@0
   141
	if (aRegion.Count() > 20 || aRegion.CheckError())
sl@0
   142
		return ETrue;
sl@0
   143
	TBool hit = EFalse;
sl@0
   144
	if (iCount > 0)
sl@0
   145
		{
sl@0
   146
		const TRect * rect = aRegion.RectangleList();
sl@0
   147
		for (TInt num = 0; num < aRegion.Count(); ++num)
sl@0
   148
			{
sl@0
   149
			hit = FillRect(*rect) || hit;
sl@0
   150
			if (iCount < 1)
sl@0
   151
				break;
sl@0
   152
			++rect;
sl@0
   153
			}
sl@0
   154
		}
sl@0
   155
	return hit;
sl@0
   156
	}
sl@0
   157
	
sl@0
   158
// x >> 5 is equivalent to x / 32
sl@0
   159
// 0x1F is the rounding error when dividing by 32
sl@0
   160
// 0x20 is 32.
sl@0
   161
// The compiler might do all the optimizations for us - not checked.
sl@0
   162
TBool CScreen::CFallbackMap::FillRect(const TRect& aRect)
sl@0
   163
	{
sl@0
   164
	TBool hit = EFalse;
sl@0
   165
	const TSizeMode & mode = iScreen->ScreenSizeModeData();
sl@0
   166
	TRect scrrect(mode.iOrigin, mode.iScreenSize);
sl@0
   167
	TRect rect = aRect;
sl@0
   168
	rect.Intersection(scrrect);
sl@0
   169
	TInt rowWidthInInts = mode.iScreenSize.iWidth;
sl@0
   170
	if (rowWidthInInts & 0x1F)
sl@0
   171
		rowWidthInInts += 0x20;
sl@0
   172
	rowWidthInInts >>= 5;
sl@0
   173
sl@0
   174
	TInt colStartInInts = rect.iTl.iX >> 5;
sl@0
   175
	TInt firstOffsetInBits = rect.iTl.iX & 0x1F;
sl@0
   176
sl@0
   177
	for(TInt row = rect.iTl.iY; row < rect.iBr.iY; ++row)
sl@0
   178
		{
sl@0
   179
		TInt * map = iMap + row * rowWidthInInts + colStartInInts;
sl@0
   180
		TInt offsetShift = 31 - firstOffsetInBits;
sl@0
   181
		for (TInt col = rect.iTl.iX; col < rect.iBr.iX; ++col)
sl@0
   182
			{
sl@0
   183
			WS_ASSERT_DEBUG(map - iMap < iMapSize, EWsPanicScreenFallback);
sl@0
   184
			if (!(*map & 1 << offsetShift))
sl@0
   185
				{
sl@0
   186
				--iCount;
sl@0
   187
				hit = ETrue;
sl@0
   188
				if (iCount < 1)
sl@0
   189
					break;
sl@0
   190
				(*map) |= (1 << offsetShift);
sl@0
   191
				}
sl@0
   192
			--offsetShift;
sl@0
   193
			if (offsetShift < 0)
sl@0
   194
				{
sl@0
   195
				offsetShift = 31;
sl@0
   196
				++map;
sl@0
   197
				}
sl@0
   198
			}
sl@0
   199
		}
sl@0
   200
	return hit;
sl@0
   201
	}
sl@0
   202
sl@0
   203
TInt CScreen::CFallbackMap::Count() const
sl@0
   204
	{
sl@0
   205
	return iCount;
sl@0
   206
	}
sl@0
   207
sl@0
   208
const TRect * CScreen::CFallbackMap::Rect() const
sl@0
   209
	{
sl@0
   210
	return iRegion.RectangleList();
sl@0
   211
	}
sl@0
   212
	
sl@0
   213
const RRegion * CScreen::CFallbackMap::Region() const
sl@0
   214
	{
sl@0
   215
	return &iRegion;
sl@0
   216
	}
sl@0
   217
sl@0
   218
TInt CScreen::CFallbackMap::Resize(const TSize& aSize)
sl@0
   219
	{
sl@0
   220
	TInt err = KErrNone;
sl@0
   221
	TInt width = (aSize.iWidth+31) >> 5;	// divide by 32
sl@0
   222
	TInt invertedWidth = (aSize.iHeight+31) >> 5;	// divide by 32
sl@0
   223
		
sl@0
   224
	TInt maxSize = Max(width * aSize.iHeight,invertedWidth * aSize.iWidth);
sl@0
   225
	if (maxSize > iMapSize)
sl@0
   226
		{
sl@0
   227
		TInt* newMap=NULL;
sl@0
   228
		newMap = new  TInt [maxSize];
sl@0
   229
		if (newMap)
sl@0
   230
			{
sl@0
   231
			delete []iMap;
sl@0
   232
			iMap = newMap;
sl@0
   233
			iMapSize = maxSize;
sl@0
   234
			}
sl@0
   235
		else
sl@0
   236
			{
sl@0
   237
			err = KErrNoMemory;
sl@0
   238
			}
sl@0
   239
		}
sl@0
   240
	return err;
sl@0
   241
	}
sl@0
   242
	
sl@0
   243
//
sl@0
   244
// CScreen
sl@0
   245
//
sl@0
   246
CScreen::CScreen(): iDirects(_FOFF(CWsDirectScreenAccess,iLink)), iMaxContrast(-1), iMaxBrightness(-1)
sl@0
   247
, iDisplayChangeSpinner(0), iConfigChangeSpinner(0)
sl@0
   248
	{
sl@0
   249
	}
sl@0
   250
sl@0
   251
CScreen::~CScreen()
sl@0
   252
	{
sl@0
   253
	delete iDebugBar;
sl@0
   254
	TInt ii;
sl@0
   255
	if(iModes)
sl@0
   256
		{
sl@0
   257
		for(ii=iNumScreenSizeModes-1;ii>=0;--ii)
sl@0
   258
			{
sl@0
   259
			delete (*iModes)[ii];
sl@0
   260
			}
sl@0
   261
		iModes->Close();
sl@0
   262
		delete iModes;
sl@0
   263
		}
sl@0
   264
	delete iRootWindow;
sl@0
   265
	iScreenDevice = NULL;
sl@0
   266
	delete iDsaDevice;
sl@0
   267
	delete iDeviceMap;
sl@0
   268
	delete iFallbackMap;
sl@0
   269
	delete iSpriteManager;
sl@0
   270
	delete iWindowElementSet;
sl@0
   271
	if (!iDsaSurface.IsNull())
sl@0
   272
	    {
sl@0
   273
	    TInt err = iScene->UnregisterSurface(iDsaSurface);
sl@0
   274
		WS_ASSERT_DEBUG(KErrNone == err, EWsPanicDirectScreenAccess);
sl@0
   275
	    }
sl@0
   276
sl@0
   277
	delete iSurfaceMap;
sl@0
   278
	delete iRedraw;
sl@0
   279
#if defined(__WINS__) && defined(_DEBUG)
sl@0
   280
	delete iDebugWin;
sl@0
   281
#endif
sl@0
   282
	
sl@0
   283
	if(iDisplayChangeNotifier)
sl@0
   284
		delete iDisplayChangeNotifier;
sl@0
   285
	if(iConfigChangeNotifier)
sl@0
   286
		delete iConfigChangeNotifier;
sl@0
   287
	iWsClientList.Close();
sl@0
   288
	}
sl@0
   289
sl@0
   290
void CScreen::ConstructL(const TRect& aDigitiserArea, TInt aScreenNumber)
sl@0
   291
	{
sl@0
   292
	iScreenNumber = aScreenNumber ;
sl@0
   293
sl@0
   294
	if (wsDebugLog)
sl@0
   295
		{
sl@0
   296
		_LIT(KWSERVInitScreen,"Initialising for Screen %d");
sl@0
   297
		wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant, KWSERVInitScreen, aScreenNumber);
sl@0
   298
		}
sl@0
   299
sl@0
   300
	// create screen redraw with render stages
sl@0
   301
	iRedraw = CScreenRedraw::NewL(*this);
sl@0
   302
	iScreenDevice = iRedraw->ObjectInterface<MWsScreenDevice>();
sl@0
   303
	WS_ASSERT_ALWAYS(iScreenDevice, EWsPanicScreenDeviceMissing);
sl@0
   304
	iScene = iRedraw->ObjectInterface<MWsScene>();
sl@0
   305
	WS_ASSERT_ALWAYS(iScene, EWsPanicSceneMissing);
sl@0
   306
	
sl@0
   307
	iDeviceMap = CGraphicsDeviceMap::NewL(*iScreenDevice);
sl@0
   308
sl@0
   309
	iDisplayControl = MWsScreen::ObjectInterface<MWsDisplayControl>();
sl@0
   310
	iDisplayMapping = MWsScreen::ObjectInterface<MWsDisplayMapping>();
sl@0
   311
	iDisplayPolicy = MWsScreen::ObjectInterface<MWsDisplayPolicy>();
sl@0
   312
sl@0
   313
sl@0
   314
	// initialize screen size mode data
sl@0
   315
	LoadScreenSizesL(iScreenDevice->SizeInPixels());
sl@0
   316
	iFallbackMap = CFallbackMap::NewL(this);
sl@0
   317
sl@0
   318
	LoadScreenSizeProperties(iScreenDevice->DisplayMode());
sl@0
   319
sl@0
   320
	if (iDisplayPolicy)
sl@0
   321
		{
sl@0
   322
		iDisplayPolicy->NewAppModesAvailable();
sl@0
   323
		}
sl@0
   324
	
sl@0
   325
	iDigitiserArea = aDigitiserArea;
sl@0
   326
	SetInitialScreenSizeModeAndRotation(); //get the first/lowest valid mode.  Also calls SetDigitiserAreas
sl@0
   327
	
sl@0
   328
	ApplyRemainingWsiniSettingsL();
sl@0
   329
	
sl@0
   330
	iRootWindow = new (ELeave) CWsRootWindow(NULL, this);
sl@0
   331
	iRootWindow->ConstructL();
sl@0
   332
	
sl@0
   333
	// Default fading parameters
sl@0
   334
	iBlackMap=128;
sl@0
   335
	iWhiteMap=255;
sl@0
   336
	
sl@0
   337
	iSpriteManager = CWsSpriteManager::NewL();
sl@0
   338
sl@0
   339
	InitializeSceneL();
sl@0
   340
	InitializeUiElementsL();
sl@0
   341
	iSurfaceMap = new (ELeave) CRegisteredSurfaceMap(*iScene);
sl@0
   342
	
sl@0
   343
	//if the interface for notification is available. start notification AO.
sl@0
   344
	if (iDisplayControl)
sl@0
   345
		{
sl@0
   346
		iDisplayChangeNotifier = CWsDisplayChangeNotifier::NewL(iDisplayControl, this);
sl@0
   347
		iDisplayChangeNotifier->IssueNotificationRequest();
sl@0
   348
		iConfigChangeNotifier = CWsConfigChangeNotifier::NewL(iDisplayControl, this);
sl@0
   349
		iConfigChangeNotifier->IssueNotificationRequest();
sl@0
   350
		}
sl@0
   351
	doSetScreenMode(iScreenSizeMode,ETrue);
sl@0
   352
	}
sl@0
   353
sl@0
   354
void CScreen::ApplyRemainingWsiniSettingsL()
sl@0
   355
	{
sl@0
   356
#if defined(__WINS__) && defined(_DEBUG)
sl@0
   357
	_LIT(KDebugOsb,"DEBUGOSB");
sl@0
   358
	if(WsIniFile->FindVar(iScreenNumber, KDebugOsb))
sl@0
   359
		{
sl@0
   360
		_LIT(KDebugWinTitleFormat, "Screen %d, DSA surface");
sl@0
   361
		TBuf<32> title;
sl@0
   362
		title.Format(KDebugWinTitleFormat, iScreenNumber);
sl@0
   363
		iDebugWin = CDebugOsbWin::NewL(title, iScreenDevice->SizeInPixels());
sl@0
   364
		}
sl@0
   365
#endif
sl@0
   366
	
sl@0
   367
   	TInt autoClear = 1;
sl@0
   368
   	_LIT(KWSERVIniFileVarAutoClear,"AUTOCLEAR");
sl@0
   369
   	WsIniFile->FindVar(iScreenNumber,KWSERVIniFileVarAutoClear,autoClear);
sl@0
   370
   	if (autoClear != 0)
sl@0
   371
   		{
sl@0
   372
   		iFlags|=EAutoClear;
sl@0
   373
   		}
sl@0
   374
		
sl@0
   375
	_LIT(KBackLight,"BACKLIGHTCONTROL");
sl@0
   376
	iBackLightFlag=WsIniFile->FindVar( iScreenNumber, KBackLight);
sl@0
   377
sl@0
   378
	_LIT(KWSERVIniFileVarBlankScreen, "BLANKSCREENONROTATION");
sl@0
   379
	if (WsIniFile->FindVar(iScreenNumber, KWSERVIniFileVarBlankScreen))
sl@0
   380
		{
sl@0
   381
		iFlags|=EBlankScreenOnRotation;
sl@0
   382
		}
sl@0
   383
sl@0
   384
	//Cache pointers to renderstage APIs required in CHANGETRACKING mode
sl@0
   385
	iWindowTreeObserver = iRedraw->ObjectInterface<MWsWindowTreeObserver>(); 
sl@0
   386
	iDrawAnnotationObserver = iRedraw->ObjectInterface<MWsDrawAnnotationObserver>();
sl@0
   387
	iWindowVisibilityNotifier = iRedraw->ObjectInterface<MWsWindowVisibilityNotifier>();
sl@0
   388
	if(iWindowVisibilityNotifier)
sl@0
   389
		{
sl@0
   390
		iWindowVisibilityNotifier->RegisterWindowVisibilityObserver(iRedraw);
sl@0
   391
		}
sl@0
   392
	if (WsIniFile->FindVar(iScreenNumber, KWSERVIniFileVarChangeTracking))
sl@0
   393
		{
sl@0
   394
		iFlags|=EChangeTracking;
sl@0
   395
		}
sl@0
   396
sl@0
   397
	//coverity[const]
sl@0
   398
	TInt refreshRate = 1000000;
sl@0
   399
	_LIT(KDebugBar, "DEBUGBAR");
sl@0
   400
	if (WsIniFile->FindVar(KDebugBar, refreshRate))
sl@0
   401
		{
sl@0
   402
		if (refreshRate < 100000)
sl@0
   403
			// coverity [dead_error_line]
sl@0
   404
			refreshRate = 50000;
sl@0
   405
		iDebugBar = CDebugBar::NewL(this, refreshRate);
sl@0
   406
		}
sl@0
   407
	}
sl@0
   408
sl@0
   409
void CScreen::AcquireDsaScreenDeviceL()
sl@0
   410
	{
sl@0
   411
	//creates WSERV's DSA buffer handle
sl@0
   412
	//registers the DSA surface into the scene accordingly to the value of aRegisterSurface
sl@0
   413
	if(!iDsaDevice)
sl@0
   414
		{
sl@0
   415
		TDisplayMode screenMode = ENone;
sl@0
   416
		screenMode = iScreenDevice->DisplayMode();
sl@0
   417
		if(screenMode != ENone)
sl@0
   418
			{
sl@0
   419
			CreateDsaScreenDeviceIfSupportedL(screenMode);
sl@0
   420
			// initialize DSA
sl@0
   421
			iDsaDevice->SetAutoUpdate(EFalse);
sl@0
   422
			iDsaDevice->SetDeviceOrientation(GcToDevice(ScreenSizeModeData().iRotation));
sl@0
   423
			iDsaDevice->ChangeScreenDevice(NULL);    //This is necessary to initialise the screen
sl@0
   424
			// register DSA Surface
sl@0
   425
			TInt err = InitializeDsaSurface();
sl@0
   426
			// create a graphics context to clear the DSA surface
sl@0
   427
			if (!err)
sl@0
   428
				err = iDsaDevice->CreateContext(iDsaGc);
sl@0
   429
			if (!err)
sl@0
   430
				iDsaGc->Activate(iDsaDevice);
sl@0
   431
			if (err != KErrNone)
sl@0
   432
				{
sl@0
   433
				//Creation of the DSA surface failed
sl@0
   434
				//Cleanup the DSA Surface ID
sl@0
   435
				iDsaSurface = TSurfaceId::CreateNullId();
sl@0
   436
				//and the iDsaDevice
sl@0
   437
				delete iDsaDevice;
sl@0
   438
				iDsaDevice = NULL;
sl@0
   439
				User::Leave(err);
sl@0
   440
				}
sl@0
   441
			}
sl@0
   442
		else
sl@0
   443
			{
sl@0
   444
			User::Leave(KErrNotSupported);
sl@0
   445
			}
sl@0
   446
		}
sl@0
   447
	iNumberDrawingDsa++;
sl@0
   448
	}
sl@0
   449
sl@0
   450
void CScreen::CreateDsaScreenDeviceIfSupportedL(TDisplayMode aScreenMode)
sl@0
   451
	{
sl@0
   452
	if(DoCreateDsaScreenDevice(aScreenMode))
sl@0
   453
		return;
sl@0
   454
	// Try creating the screen device with all available display modes, going from best to worst
sl@0
   455
	__ASSERT_COMPILE(EColorLast == 14); // if any display mode is added to TDisplayMode we must update the list below
sl@0
   456
	// (the list below contains all enums in TDisplayMode except ENone, ERgb, EColorLast)
sl@0
   457
	if(DoCreateDsaScreenDevice(EColor16MAP))
sl@0
   458
		return;
sl@0
   459
	if(DoCreateDsaScreenDevice(EColor16MA))
sl@0
   460
		return;
sl@0
   461
	if(DoCreateDsaScreenDevice(EColor16MU))
sl@0
   462
		return;
sl@0
   463
	if(DoCreateDsaScreenDevice(EColor16M))
sl@0
   464
		return;
sl@0
   465
	if(DoCreateDsaScreenDevice(EColor64K))
sl@0
   466
		return;
sl@0
   467
	if(DoCreateDsaScreenDevice(EColor4K))
sl@0
   468
		return;
sl@0
   469
	if(DoCreateDsaScreenDevice(EColor256))
sl@0
   470
		return;
sl@0
   471
	if(DoCreateDsaScreenDevice(EColor16))
sl@0
   472
		return;
sl@0
   473
	if(DoCreateDsaScreenDevice(EGray256))
sl@0
   474
		return;
sl@0
   475
	if(DoCreateDsaScreenDevice(EGray16))
sl@0
   476
		return;
sl@0
   477
	if(DoCreateDsaScreenDevice(EGray4))
sl@0
   478
		return;
sl@0
   479
	if(DoCreateDsaScreenDevice(EGray2))
sl@0
   480
		return;
sl@0
   481
	User::Leave(KErrNotSupported);
sl@0
   482
	}
sl@0
   483
sl@0
   484
TBool CScreen::DoCreateDsaScreenDevice(TDisplayMode aScreenMode)
sl@0
   485
	{
sl@0
   486
	TRAPD(err, iDsaDevice = CFbsScreenDevice::NewL(iScreenNumber, aScreenMode));
sl@0
   487
	if(err == KErrNone)
sl@0
   488
		{
sl@0
   489
		TUint supportedDsaRotationModes = iDsaDevice->DeviceOrientationsAvailable();
sl@0
   490
		MWsScene::TSceneRotation currenTSceneRotation = iScene->SceneRotation();
sl@0
   491
		TBool doesDsaSupportThisMode = EFalse;
sl@0
   492
		switch(currenTSceneRotation)
sl@0
   493
			{
sl@0
   494
			case MWsScene::ESceneAntiClockwise0:
sl@0
   495
				if(supportedDsaRotationModes & EDeviceOrientationNormal)
sl@0
   496
					{
sl@0
   497
					doesDsaSupportThisMode = ETrue;
sl@0
   498
					}
sl@0
   499
				break;
sl@0
   500
			case MWsScene::ESceneAntiClockwise90:
sl@0
   501
				if(supportedDsaRotationModes & EDeviceOrientation90CW)
sl@0
   502
					{
sl@0
   503
					doesDsaSupportThisMode = ETrue;
sl@0
   504
					}
sl@0
   505
				break;				
sl@0
   506
			case MWsScene::ESceneAntiClockwise180:
sl@0
   507
				if(supportedDsaRotationModes & EDeviceOrientation180)
sl@0
   508
					{
sl@0
   509
					doesDsaSupportThisMode = ETrue;
sl@0
   510
					}
sl@0
   511
				break;				
sl@0
   512
			case MWsScene::ESceneAntiClockwise270:
sl@0
   513
				if(supportedDsaRotationModes & EDeviceOrientation270CW)
sl@0
   514
					{
sl@0
   515
					doesDsaSupportThisMode = ETrue;
sl@0
   516
					}
sl@0
   517
				break;				
sl@0
   518
			default:
sl@0
   519
				RDebug::Print(_L("** CScreen::DoCreateDsaScreenDevice Panic, non existing rotation mode"));
sl@0
   520
				WS_PANIC_ALWAYS(EWsPanicInvalidOperation);
sl@0
   521
				break;
sl@0
   522
			}
sl@0
   523
		if(!doesDsaSupportThisMode)
sl@0
   524
			{
sl@0
   525
			delete iDsaDevice;
sl@0
   526
			iDsaDevice = NULL;
sl@0
   527
			RDebug::Print(_L("** Current Rotation Mode not supported by the DSA device"));
sl@0
   528
			err = KErrNotSupported;
sl@0
   529
			}
sl@0
   530
		}
sl@0
   531
	return (err == KErrNone);
sl@0
   532
	}
sl@0
   533
sl@0
   534
void CScreen::AbortDSAs(RDirectScreenAccess::TTerminationReasons aReason, TSglQue<CWsDirectScreenAccess>& aDirects)
sl@0
   535
	{
sl@0
   536
	if (aDirects.IsEmpty())
sl@0
   537
		return;
sl@0
   538
sl@0
   539
	TInt nofDSAs = 0;
sl@0
   540
	CWsDirectScreenAccess* direct= NULL;
sl@0
   541
	TSglQueIter<CWsDirectScreenAccess> iter(aDirects);
sl@0
   542
	while ((direct=iter++)!=NULL)
sl@0
   543
		{
sl@0
   544
		nofDSAs++;
sl@0
   545
		direct->SignalAbort(aReason);
sl@0
   546
		}
sl@0
   547
sl@0
   548
	TRequestStatus timerStatus;
sl@0
   549
	RTimer& timer=CWsTop::Timer();
sl@0
   550
	timer.Cancel();
sl@0
   551
sl@0
   552
	TRequestStatus** cancelReqList = (TRequestStatus**) User::AllocZ(sizeof(TRequestStatus*) * (nofDSAs + 1));
sl@0
   553
	if (NULL != cancelReqList)
sl@0
   554
		{
sl@0
   555
		TInt dsaNo = 1;
sl@0
   556
		timer.After(timerStatus, KDSAAbortingImmediateRespAwaitFrameMicrosec);
sl@0
   557
		iter.SetToFirst();
sl@0
   558
		while ((direct=iter++)!=NULL)
sl@0
   559
			{
sl@0
   560
			WS_ASSERT_DEBUG((dsaNo<=(nofDSAs)),EWsPanicDirectScreenAccess);
sl@0
   561
			cancelReqList[ dsaNo ] = &direct->AbortStatus();
sl@0
   562
			dsaNo++;
sl@0
   563
			}
sl@0
   564
		cancelReqList[ 0 ] = &timerStatus;
sl@0
   565
sl@0
   566
		//wait for response or timeout
sl@0
   567
		User::WaitForNRequest(cancelReqList, nofDSAs + 1);
sl@0
   568
sl@0
   569
		iter.SetToFirst();
sl@0
   570
		while ((direct=iter++)!=NULL)
sl@0
   571
			{
sl@0
   572
			if (direct->AbortStatus() != KRequestPending)
sl@0
   573
				direct->CancelAbortObject(); // responded
sl@0
   574
			else
sl@0
   575
				direct->Abort();
sl@0
   576
			}
sl@0
   577
sl@0
   578
		if (timerStatus == KRequestPending)
sl@0
   579
			{
sl@0
   580
			timer.Cancel();
sl@0
   581
			User::WaitForRequest(timerStatus);
sl@0
   582
			}
sl@0
   583
sl@0
   584
		User::Free(cancelReqList);
sl@0
   585
		}
sl@0
   586
	else
sl@0
   587
		{
sl@0
   588
		iter.SetToFirst();
sl@0
   589
		while ((direct=iter++) != NULL)
sl@0
   590
			{
sl@0
   591
			TRequestStatus timerStatus;
sl@0
   592
			RTimer& timer=CWsTop::Timer();
sl@0
   593
			timer.Cancel();
sl@0
   594
			timer.After(timerStatus, KDSAAbortingImmediateRespAwaitFrameMicrosec);
sl@0
   595
			
sl@0
   596
			//wait for response or timeout
sl@0
   597
			User::WaitForRequest(direct->AbortStatus(), timerStatus);
sl@0
   598
			
sl@0
   599
			if (direct->AbortStatus() != KRequestPending)
sl@0
   600
				direct->CancelAbortObject(); //responded
sl@0
   601
			else
sl@0
   602
				direct->Abort(); //timed out
sl@0
   603
sl@0
   604
			if (timerStatus == KRequestPending)
sl@0
   605
				{
sl@0
   606
				timer.Cancel();
sl@0
   607
				User::WaitForRequest(timerStatus);
sl@0
   608
				}
sl@0
   609
			}
sl@0
   610
		}
sl@0
   611
	}
sl@0
   612
sl@0
   613
void CScreen::AbortAllDirectDrawing(RDirectScreenAccess::TTerminationReasons aReason)
sl@0
   614
	{
sl@0
   615
	AbortDSAs(aReason,iDirects);
sl@0
   616
	}
sl@0
   617
sl@0
   618
void CScreen::AddDirect(CWsDirectScreenAccess& aDirect)
sl@0
   619
	{
sl@0
   620
	TBool emptyBefore = iDirects.IsEmpty();
sl@0
   621
	iDirects.AddLast(aDirect);
sl@0
   622
	TBool emptyAfter = iDirects.IsEmpty();
sl@0
   623
	if (emptyBefore && ! emptyAfter)
sl@0
   624
		{
sl@0
   625
		TWsEvent wsevent;
sl@0
   626
		wsevent.SetType(EEventDirectScreenAccessBegin);
sl@0
   627
		*(wsevent.Int()) = iScreenNumber;
sl@0
   628
		TWindowServerEvent::PublishNotification(wsevent);
sl@0
   629
		}
sl@0
   630
sl@0
   631
	if (iDsaDrawState==EDsaDrawStateIdle && aDirect.IsVisible())
sl@0
   632
		{
sl@0
   633
		iDsaDrawState = EDsaDrawStateDrawing;
sl@0
   634
		TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EDsaDrawingBegin, iScreenNumber));
sl@0
   635
		}
sl@0
   636
	}
sl@0
   637
sl@0
   638
void CScreen::RemoveDirect(CWsDirectScreenAccess& aDirect)
sl@0
   639
	{
sl@0
   640
	TBool emptyBefore = iDirects.IsEmpty();
sl@0
   641
	iDirects.Remove(aDirect);
sl@0
   642
	TBool emptyAfter = iDirects.IsEmpty();
sl@0
   643
	if (emptyAfter && ! emptyBefore)
sl@0
   644
		{
sl@0
   645
		TWsEvent wsevent;
sl@0
   646
		wsevent.SetType(EEventDirectScreenAccessEnd);
sl@0
   647
		*(wsevent.Int()) = iScreenNumber;
sl@0
   648
		TWindowServerEvent::PublishNotification(wsevent);
sl@0
   649
		}
sl@0
   650
sl@0
   651
	if (iDsaDrawState==EDsaDrawStateDrawing && aDirect.IsVisible() && !HasVisibleDirectOnQueue())
sl@0
   652
		{
sl@0
   653
		iDsaDrawState = EDsaDrawStateIdle;
sl@0
   654
		TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EDsaDrawingEnd, iScreenNumber));
sl@0
   655
		}
sl@0
   656
	}
sl@0
   657
sl@0
   658
TBool CScreen::HasVisibleDirectOnQueue()
sl@0
   659
	{
sl@0
   660
	if (iDirects.IsEmpty())
sl@0
   661
		return EFalse;
sl@0
   662
sl@0
   663
	TSglQueIter<CWsDirectScreenAccess> iter(iDirects);
sl@0
   664
	CWsDirectScreenAccess* dsa;
sl@0
   665
	while ((dsa=iter++)!=NULL)
sl@0
   666
		{
sl@0
   667
		if (dsa->IsVisible())
sl@0
   668
			return ETrue;
sl@0
   669
		}
sl@0
   670
sl@0
   671
	return EFalse;
sl@0
   672
	}
sl@0
   673
sl@0
   674
#if defined(_DEBUG)
sl@0
   675
TBool CScreen::IsDirectOnQueue(const CWsDirectScreenAccess* aDirect)
sl@0
   676
	{
sl@0
   677
	TSglQueIter<CWsDirectScreenAccess> iter(iDirects);
sl@0
   678
	CWsDirectScreenAccess* direct;
sl@0
   679
	while ((direct=iter++)!=NULL)
sl@0
   680
		{
sl@0
   681
		if (direct==aDirect)
sl@0
   682
			return ETrue;
sl@0
   683
		}
sl@0
   684
	return EFalse;
sl@0
   685
	}
sl@0
   686
#endif
sl@0
   687
sl@0
   688
void CScreen::KillForegroundSession()
sl@0
   689
	{
sl@0
   690
	if (iCurrentFocus)
sl@0
   691
		{
sl@0
   692
		_LIT(KWSERVKillWinGp,"Killing Session owning Window Group with Id=%d");
sl@0
   693
		if (wsDebugLog)
sl@0
   694
			wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVKillWinGp,iCurrentFocus->Identifier());
sl@0
   695
		iCurrentFocus->WsOwner()->SessionTerminate();
sl@0
   696
		}
sl@0
   697
	}
sl@0
   698
sl@0
   699
CWsWindowGroup* CScreen::FindNewFocus(CWsRootWindow* aRootWindow)
sl@0
   700
	{
sl@0
   701
	CWsWindowGroup* newFocus;
sl@0
   702
	for(newFocus=aRootWindow->Child();newFocus && newFocus->CanReceiveFocus()==EFalse;newFocus=newFocus->NextSibling()) {}
sl@0
   703
sl@0
   704
	return newFocus;
sl@0
   705
	}
sl@0
   706
sl@0
   707
void CScreen::ResetFocus(CWsWindowGroup* aClosingWindow)
sl@0
   708
	{
sl@0
   709
	CWsWindowGroup* oldFocus=iCurrentFocus;
sl@0
   710
	CWsWindowGroup* newFocus=NULL;
sl@0
   711
	CScreen* newFocusedScreen=NULL;
sl@0
   712
	iCurrentFocus=FindNewFocus(iRootWindow);
sl@0
   713
	TBool focusedScreen= EFalse;
sl@0
   714
	/*Focus policy is specified in the wsini.ini file using the keyword 'MULTIFOCUSPOLICY'.
sl@0
   715
	If the keyword is not specified, then the default policy is run.
sl@0
   716
	*/
sl@0
   717
	if(!CWsTop::MultiFocusPolicy())
sl@0
   718
		{
sl@0
   719
		focusedScreen=(this==CWsTop::CurrentFocusScreen()); //check if this screen is the focus screen
sl@0
   720
		if (!iCurrentFocus && focusedScreen)
sl@0
   721
			{
sl@0
   722
			/*If this screen is the focused screen but does not have a focusable window group, then search for the
sl@0
   723
			next screen that has a focusable window group and set that screen as the focused screen.
sl@0
   724
			*/
sl@0
   725
			CScreen* screen=NULL;
sl@0
   726
			TInt screenNo;
sl@0
   727
			for (screenNo=0; screenNo<CWsTop::NumberOfScreens() && !newFocus; ++screenNo)
sl@0
   728
				{
sl@0
   729
				if (screenNo!=iScreenNumber)
sl@0
   730
					{
sl@0
   731
					screen=CWsTop::Screen(screenNo);
sl@0
   732
					newFocus=FindNewFocus(screen->RootWindow());
sl@0
   733
					}
sl@0
   734
				}
sl@0
   735
			if (newFocus)
sl@0
   736
				newFocusedScreen=screen;
sl@0
   737
			}
sl@0
   738
		}
sl@0
   739
	/*Scenario A: multi-focus policy
sl@0
   740
			newFocusedScreen is NULL
sl@0
   741
			focusedScreen is EFalse
sl@0
   742
			CWsTop::MultiFocusPolicy() returns ETrue
sl@0
   743
			Check if the new focusable window group is not the same, send focus lost message to window group
sl@0
   744
			that has just lost focus and send focus gain message to window group that can receive focus.
sl@0
   745
	  Scenario B: single-focus policy (default)
sl@0
   746
			CWsTop::MultiFocusPolicy() returns EFalse
sl@0
   747
			Check if the new focusable window group is not the same or if there is a new focused screen, send focus lost
sl@0
   748
			message to window group that has just lost focus and send focus gain message to window group that can receive focus.
sl@0
   749
	*/
sl@0
   750
	if (iCurrentFocus!=oldFocus || newFocusedScreen)
sl@0
   751
		{
sl@0
   752
		if (oldFocus && (focusedScreen || CWsTop::MultiFocusPolicy()) && oldFocus!=aClosingWindow)
sl@0
   753
			{
sl@0
   754
			oldFocus->LostFocus();
sl@0
   755
			}
sl@0
   756
		if (newFocusedScreen)
sl@0
   757
			{
sl@0
   758
			CWsTop::SetCurrentFocusScreen(newFocusedScreen);
sl@0
   759
			newFocus->ReceivedFocus();
sl@0
   760
			}
sl@0
   761
		else if (iCurrentFocus && (focusedScreen || CWsTop::MultiFocusPolicy()))
sl@0
   762
			{
sl@0
   763
			iCurrentFocus->ReceivedFocus();
sl@0
   764
			}
sl@0
   765
		TWsPointer::UpdatePointerCursor();
sl@0
   766
		TWindowServerEvent::SendFocusChangedEvents();
sl@0
   767
		}
sl@0
   768
	TWindowServerEvent::SendGroupListChangedEvents();
sl@0
   769
	}
sl@0
   770
sl@0
   771
void CScreen::RemoveFromDefaultOwningList(CWsWindowGroup *aDestroyedGroup)
sl@0
   772
	{
sl@0
   773
	for (CWsWindowGroup **group=&iDefaultOwningWindow;*group;group=(*group)->NextDefaultOwningWindowPtr())
sl@0
   774
		{
sl@0
   775
		if (*group==aDestroyedGroup)
sl@0
   776
			{
sl@0
   777
			*group=*aDestroyedGroup->NextDefaultOwningWindowPtr();
sl@0
   778
			break;
sl@0
   779
			}
sl@0
   780
		}
sl@0
   781
	}
sl@0
   782
sl@0
   783
void CScreen::SetDefaultOwningWindow(CWsWindowGroup *aGroup)
sl@0
   784
	{
sl@0
   785
	RemoveFromDefaultOwningList(aGroup);
sl@0
   786
	aGroup->SetNextDefaultOwningWindow(iDefaultOwningWindow);
sl@0
   787
	iDefaultOwningWindow=aGroup;
sl@0
   788
	}
sl@0
   789
sl@0
   790
void CScreen::GetScanLine(const TWsSdCmdGetScanLine *aGetScanLine)
sl@0
   791
	{
sl@0
   792
	TRgb buf[EGetScanLineBufLen];
sl@0
   793
	TPtr8 des((TUint8 *)&buf[0],EGetScanLineBufLen*sizeof(TRgb));
sl@0
   794
	TPoint pos(aGetScanLine->pos);
sl@0
   795
	TInt read=0;
sl@0
   796
	TInt len=(des.MaxLength()*EGetScanLineBufLen)/CFbsBitmap::ScanLineLength(EGetScanLineBufLen,aGetScanLine->dispMode);
sl@0
   797
	if (aGetScanLine->len < 0 || (CFbsBitmap::ScanLineLength(aGetScanLine->len, aGetScanLine->dispMode) >
sl@0
   798
									CWsClient::CurrentClient()->ClientMessage().GetDesMaxLength(1)))
sl@0
   799
		{
sl@0
   800
		CWsClient::PanicCurrentClient(EWservPanicInvalidParameter);
sl@0
   801
		}
sl@0
   802
	FOREVER
sl@0
   803
		{
sl@0
   804
		if ((aGetScanLine->len-read)<len)
sl@0
   805
			len=aGetScanLine->len-read;
sl@0
   806
		iScreenDevice->GetScanLine(des,pos,len,aGetScanLine->dispMode);
sl@0
   807
		CWsClient::ReplyBuf(des);
sl@0
   808
		read+=len;
sl@0
   809
		if (read==aGetScanLine->len)
sl@0
   810
			break;
sl@0
   811
		pos.iX+=len;
sl@0
   812
		}
sl@0
   813
	}
sl@0
   814
sl@0
   815
void CScreen::MaxNumColors(TInt& aColors,TInt& aGrays)
sl@0
   816
	{
sl@0
   817
	aGrays=0;
sl@0
   818
	aColors=TDisplayModeUtils::NumDisplayModeColors(DisplayMode());
sl@0
   819
	if (!TDisplayModeUtils::IsDisplayModeColor(DisplayMode()))
sl@0
   820
		{
sl@0
   821
		aGrays=aColors;
sl@0
   822
		aColors=0;
sl@0
   823
		}
sl@0
   824
	}
sl@0
   825
sl@0
   826
#define MODE_TO_FLAG(x) 1<<(x-1)
sl@0
   827
#define ROTATION_TO_FLAG(x) 1<<x
sl@0
   828
const TUint allRotationsMask = ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationNormal)
sl@0
   829
						| ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated90)
sl@0
   830
						| ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated180)
sl@0
   831
						| ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated270);
sl@0
   832
const TUint KRotation0_180Mask = ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationNormal)
sl@0
   833
						| ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated180);
sl@0
   834
const TUint KRotation90_270Mask = ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated90)
sl@0
   835
						| ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated270);
sl@0
   836
TInt CScreen::ColorModesFlag()
sl@0
   837
	{
sl@0
   838
	return MODE_TO_FLAG(DisplayMode());
sl@0
   839
	}
sl@0
   840
sl@0
   841
void CScreen::UpdateDsa()
sl@0
   842
	{
sl@0
   843
	if(iDsaDevice)
sl@0
   844
		{
sl@0
   845
		#if defined(__WINS__) && defined(_DEBUG)
sl@0
   846
			if (iDebugWin)
sl@0
   847
				iDebugWin->Refresh(iDsaDevice->SizeInPixels(), iDsaDevice->DisplayMode(), iDsaDevice->Bits());
sl@0
   848
		#endif
sl@0
   849
		
sl@0
   850
		iDsaDevice->Update();
sl@0
   851
		}
sl@0
   852
	}
sl@0
   853
sl@0
   854
const CGraphicsDeviceMap& CScreen::DeviceMap() const
sl@0
   855
	{
sl@0
   856
	return *iDeviceMap;
sl@0
   857
	}
sl@0
   858
sl@0
   859
const MWsScreenDevice& CScreen::ScreenDevice() const
sl@0
   860
	{
sl@0
   861
	return *iScreenDevice; 
sl@0
   862
	}
sl@0
   863
sl@0
   864
void CScreen::SetPointerCursorArea(TInt aMode,const TRect& aRect)
sl@0
   865
	{
sl@0
   866
	(*iModes)[aMode]->iPointerCursorArea=aRect;
sl@0
   867
	(*iModes)[aMode]->iFlags |= EClientDefinedDigitiserArea;
sl@0
   868
	TWsPointer::SetPointerCursorPos(TWsPointer::PointerCursorPos());
sl@0
   869
	}
sl@0
   870
sl@0
   871
CFbsBitGc::TGraphicsOrientation CScreen::Orientation() const
sl@0
   872
	{
sl@0
   873
  	WS_ASSERT_DEBUG(IsValidScreenSizeMode(iScreenSizeMode),EWsPanicInvalidScreenSizeMode);
sl@0
   874
	return (*iModes)[iScreenSizeMode]->iRotation;
sl@0
   875
	}
sl@0
   876
sl@0
   877
TRect CScreen::DrawableArea() const
sl@0
   878
	{
sl@0
   879
	TRect drawRect=iScreenDevice->SizeInPixels();
sl@0
   880
	if (iDisplayMapping)
sl@0
   881
		{
sl@0
   882
		iDisplayMapping->MapCoordinates(EFullScreenSpace,drawRect,EApplicationSpace,drawRect);                   
sl@0
   883
		}
sl@0
   884
	return drawRect;
sl@0
   885
	}
sl@0
   886
sl@0
   887
TClientPanic CScreen::SetModeRotation(TInt aMode,CFbsBitGc::TGraphicsOrientation aRotation)
sl@0
   888
	{
sl@0
   889
	if (!IsValidScreenSizeMode(aMode))
sl@0
   890
		return EWservPanicScreenModeNumber;
sl@0
   891
	TSizeMode& mode=*(*iModes)[aMode];
sl@0
   892
	if (!(ROTATION_TO_FLAG(aRotation)&mode.iAlternativeRotations))
sl@0
   893
		return EWservPanicRotation;
sl@0
   894
	CFbsBitGc::TGraphicsOrientation oldRotation=mode.iRotation;
sl@0
   895
	mode.iRotation=aRotation;
sl@0
   896
	CWsWindowGroup::NewOrientation(aMode,aRotation, iRootWindow);
sl@0
   897
	if (aMode==ScreenSizeMode())
sl@0
   898
		{
sl@0
   899
		if(iDisplayPolicy && iDisplayControl && (mode.iAlternativeRotations == allRotationsMask))
sl@0
   900
			{
sl@0
   901
			//square mode supports all 4 rotations. We'd better do a complete Setconfiguration
sl@0
   902
			//all parameters are recalculated for 90 degree rotation
sl@0
   903
			//The most important one is the offset, since we need to re-center the appmode and it's a policy behaviour
sl@0
   904
			TDisplayConfiguration config;
sl@0
   905
			iDisplayControl->GetConfiguration(config);
sl@0
   906
			//update rotation
sl@0
   907
			config.SetRotation((TDisplayConfiguration::TRotation)aRotation);
sl@0
   908
			SetConfiguration(config);
sl@0
   909
			}
sl@0
   910
		else if (!UpdateOrientation())
sl@0
   911
			{
sl@0
   912
			// Roll back earlier change
sl@0
   913
			mode.iRotation=oldRotation;
sl@0
   914
			CWsWindowGroup::NewOrientation(aMode, oldRotation, iRootWindow);
sl@0
   915
			}
sl@0
   916
		}
sl@0
   917
	return EWservNoPanic;
sl@0
   918
	}
sl@0
   919
sl@0
   920
void CScreen::CycleDisplaySize()
sl@0
   921
	{
sl@0
   922
	TInt newMode = iScreenSizeMode;
sl@0
   923
	TSizeMode* sizeMode = NULL;
sl@0
   924
	do
sl@0
   925
		{
sl@0
   926
		newMode = (newMode+1)%iModes->Count();
sl@0
   927
		sizeMode = (*iModes)[newMode];
sl@0
   928
		}
sl@0
   929
	while (sizeMode==NULL);
sl@0
   930
	doSetScreenMode(newMode);
sl@0
   931
	}
sl@0
   932
sl@0
   933
void CScreen::doSetScreenMode(TInt aMode,TBool aInsideStartup)
sl@0
   934
	{
sl@0
   935
  	WS_ASSERT_DEBUG(IsValidScreenSizeMode(aMode),EWsPanicInvalidScreenSizeMode);
sl@0
   936
 	TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EScreenSizeModeAboutToChange, aMode));
sl@0
   937
	
sl@0
   938
	if (iDisplayControl && iDisplayPolicy)
sl@0
   939
		{
sl@0
   940
		TDisplayConfiguration config;
sl@0
   941
		TRect sizeModePosition;
sl@0
   942
		TInt error;
sl@0
   943
		TSizeMode& mode=*(*iModes)[aMode];
sl@0
   944
		config.SetRotation((TDisplayConfiguration::TRotation)mode.iRotation);
sl@0
   945
		//This will return suitable config when display is connected
sl@0
   946
		//and UI size equal to smallest appmode when disconnected
sl@0
   947
		error = iDisplayPolicy->GetSizeModeConfiguration(aMode,config,sizeModePosition);
sl@0
   948
		//set appmode in policy
sl@0
   949
		if (iDisplayMapping)
sl@0
   950
			{
sl@0
   951
			iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
sl@0
   952
			}
sl@0
   953
		MWsScene::TSceneRotation oldRotation;
sl@0
   954
		TSize newUiSize;
sl@0
   955
		config.GetResolution(newUiSize);
sl@0
   956
		if (error == KErrNone)
sl@0
   957
			{
sl@0
   958
			oldRotation = iScene->SceneRotation();
sl@0
   959
			//The config info will always be set in policy. If display is connected, policy will have
sl@0
   960
			//correct composition, Ui and app res. otherwise Ui and appmode will be both set to smallest
sl@0
   961
			//app mode, composition will be set to zero
sl@0
   962
			if (iFlags&EHasDynamicSizeModes)
sl@0
   963
				{
sl@0
   964
				error = iFallbackMap->Resize(newUiSize);
sl@0
   965
				}
sl@0
   966
			if (error == KErrNone)
sl@0
   967
				{
sl@0
   968
				error = iDisplayControl->SetConfiguration(config);
sl@0
   969
				}
sl@0
   970
			}
sl@0
   971
		if (error == KErrNone)
sl@0
   972
			{
sl@0
   973
			UpdateDynamicScreenModes();
sl@0
   974
			AbortAllDirectDrawing(RDirectScreenAccess::ETerminateScreenMode);
sl@0
   975
sl@0
   976
			if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
sl@0
   977
				{
sl@0
   978
				iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
sl@0
   979
				}
sl@0
   980
sl@0
   981
			iScreenSizeMode=aMode;
sl@0
   982
			//This could fail (to set orientation on the context, or to register the rotated DSA).
sl@0
   983
			//Previously, the update rotation was deferred, leaving the size mode out of step with the actual rotation
sl@0
   984
			//It also returns false if the orientation is "intentionally" not modified.
sl@0
   985
			(void) UpdateOrientation(&oldRotation);	
sl@0
   986
			
sl@0
   987
			//SetDigitiserAreas needs revisiting if/when we support dynamic resolutions
sl@0
   988
            //on a screen with touch input.
sl@0
   989
            //SetDigitiserAreas(newUiSize);
sl@0
   990
			
sl@0
   991
			CWsWindowGroup::SetScreenDeviceValidStates(this);
sl@0
   992
			
sl@0
   993
			if (!aInsideStartup)
sl@0
   994
				{
sl@0
   995
				iWindowElementSet->ResubmitAllElementExtents();
sl@0
   996
				//TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
sl@0
   997
				if(BlankScreenOnRotation())
sl@0
   998
					{
sl@0
   999
					iRootWindow->ClearDisplay();
sl@0
  1000
					}
sl@0
  1001
	
sl@0
  1002
				CWsTop::ClearAllRedrawStores();
sl@0
  1003
				DiscardAllSchedules();
sl@0
  1004
				iRootWindow->InvalidateWholeScreen();
sl@0
  1005
				}
sl@0
  1006
			}
sl@0
  1007
		}
sl@0
  1008
	else
sl@0
  1009
		{
sl@0
  1010
		if (iDisplayMapping)
sl@0
  1011
			{
sl@0
  1012
			TRect sizeModePosition;
sl@0
  1013
			TRAPD(err,sizeModePosition=TRect(OriginL(aMode),ScreenModeSizeInPixelsL(aMode)));
sl@0
  1014
			if (err==KErrNone)
sl@0
  1015
				{
sl@0
  1016
				iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
sl@0
  1017
				}
sl@0
  1018
			}
sl@0
  1019
		if (!aInsideStartup && (*iModes)[aMode]->iOrigin != (*iModes)[iScreenSizeMode]->iOrigin)
sl@0
  1020
			{
sl@0
  1021
			iWindowElementSet->ResubmitAllElementExtents();
sl@0
  1022
			if ((*iModes)[aMode]->iRotation == (*iModes)[iScreenSizeMode]->iRotation)
sl@0
  1023
				{
sl@0
  1024
				//TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
sl@0
  1025
				if(BlankScreenOnRotation())
sl@0
  1026
					{
sl@0
  1027
					iRootWindow->ClearDisplay();
sl@0
  1028
					}
sl@0
  1029
	
sl@0
  1030
				CWsTop::ClearAllRedrawStores();
sl@0
  1031
				DiscardAllSchedules();
sl@0
  1032
				iRootWindow->InvalidateWholeScreen();
sl@0
  1033
				}
sl@0
  1034
			}
sl@0
  1035
		iScreenSizeMode=aMode;
sl@0
  1036
		//This could fail (to set orientation on the context, or to register the rotated DSA).
sl@0
  1037
		//Previously, the update rotation was deferred, leaving the size mode out of step with the actual rotation
sl@0
  1038
		//It also returns false if the orientation is not modified.
sl@0
  1039
		(void)UpdateOrientation();
sl@0
  1040
		CWsWindowGroup::SetScreenDeviceValidStates(this);
sl@0
  1041
		}
sl@0
  1042
	TWindowServerEvent::SendScreenDeviceChangedEvents(this);
sl@0
  1043
	ResetFocus(NULL);
sl@0
  1044
	}
sl@0
  1045
sl@0
  1046
void CScreen::CycleOrientation()
sl@0
  1047
	{
sl@0
  1048
  	WS_ASSERT_DEBUG(IsValidScreenSizeMode(iScreenSizeMode),EWsPanicInvalidScreenSizeMode);
sl@0
  1049
	TSizeMode& currentSizeMode=*(*iModes)[iScreenSizeMode];
sl@0
  1050
	TUint rotations=currentSizeMode.iAlternativeRotations;
sl@0
  1051
	TInt currentRotation=currentSizeMode.iRotation;
sl@0
  1052
	TInt rotation=currentRotation+1;
sl@0
  1053
	while (rotation!=currentRotation)
sl@0
  1054
		{
sl@0
  1055
		if (rotation>CFbsBitGc::EGraphicsOrientationRotated270)
sl@0
  1056
			rotation=CFbsBitGc::EGraphicsOrientationNormal;
sl@0
  1057
		if (ROTATION_TO_FLAG(rotation)&rotations)
sl@0
  1058
			break;
sl@0
  1059
		++rotation;
sl@0
  1060
		}
sl@0
  1061
	if (rotation==currentRotation)
sl@0
  1062
		{
sl@0
  1063
		if (rotation>CFbsBitGc::EGraphicsOrientationRotated90)
sl@0
  1064
			rotation-=2;
sl@0
  1065
		else
sl@0
  1066
			rotation+=2;
sl@0
  1067
		}
sl@0
  1068
	currentSizeMode.iRotation=REINTERPRET_CAST(CFbsBitGc::TGraphicsOrientation&,rotation);
sl@0
  1069
	CWsWindowGroup::NewOrientation(iScreenSizeMode,currentSizeMode.iRotation, iRootWindow);
sl@0
  1070
sl@0
  1071
	if (!UpdateOrientation())
sl@0
  1072
		{
sl@0
  1073
		// Roll back earlier changes
sl@0
  1074
		currentSizeMode.iRotation=REINTERPRET_CAST(CFbsBitGc::TGraphicsOrientation&,currentRotation);
sl@0
  1075
		CWsWindowGroup::NewOrientation(iScreenSizeMode, currentSizeMode.iRotation, iRootWindow);
sl@0
  1076
		}
sl@0
  1077
	}
sl@0
  1078
sl@0
  1079
/**
sl@0
  1080
 * This method is called either when switching screen size mode, or when the 
sl@0
  1081
 * orientation of the currently active screen size mode is changed.
sl@0
  1082
 */
sl@0
  1083
TBool CScreen::UpdateOrientation(MWsScene::TSceneRotation* aOldRotation)
sl@0
  1084
	{
sl@0
  1085
	CFbsBitGc::TGraphicsOrientation gcOrientation = Orientation();
sl@0
  1086
sl@0
  1087
	MWsScene::TSceneRotation oldRotation = aOldRotation? (*aOldRotation):(iScene->SceneRotation());
sl@0
  1088
	MWsScene::TSceneRotation newRotation = GcToScreen(gcOrientation);
sl@0
  1089
	TDeviceOrientation newDeviceOrientation = GcToDevice(gcOrientation);
sl@0
  1090
sl@0
  1091
	// We have to disable the text cursor here while we are still in the
sl@0
  1092
	// same orientation or offset as we drew it.
sl@0
  1093
	RWsTextCursor* cursor = CWsTop::CurrentTextCursor();
sl@0
  1094
	if (cursor)
sl@0
  1095
		{
sl@0
  1096
		cursor->Disable();
sl@0
  1097
		}
sl@0
  1098
	
sl@0
  1099
// Some of this method has to be done when changing mode even if not changing rotation
sl@0
  1100
	TBool rotating=(oldRotation != newRotation);
sl@0
  1101
	if (rotating)
sl@0
  1102
		{
sl@0
  1103
	
sl@0
  1104
		// Set the new screen rotation and update the UI element extent
sl@0
  1105
		if (iScene->SetSceneRotation(newRotation) != KErrNone)
sl@0
  1106
			return EFalse;
sl@0
  1107
		// Set the new orientation for the DSA device and update the DSA surface
sl@0
  1108
		if(iDsaDevice)
sl@0
  1109
			{
sl@0
  1110
			iDsaDevice->SetDeviceOrientation(newDeviceOrientation);
sl@0
  1111
	
sl@0
  1112
			TSurfaceId newSurface;
sl@0
  1113
			iDsaDevice->GetSurface(newSurface);
sl@0
  1114
			TInt errRegister = iScene->RegisterSurface(newSurface);
sl@0
  1115
			WS_ASSERT_DEBUG(KErrNone == errRegister, EWsPanicDirectScreenAccess);
sl@0
  1116
			// This will remove all the DSA elements from the scene
sl@0
  1117
            AbortAllDirectDrawing(RDirectScreenAccess::ETerminateRotation);
sl@0
  1118
	
sl@0
  1119
			TInt errUnregister = iScene->UnregisterSurface(iDsaSurface);
sl@0
  1120
			WS_ASSERT_DEBUG(KErrNone == errUnregister, EWsPanicDirectScreenAccess);
sl@0
  1121
			iDsaSurface = newSurface;
sl@0
  1122
			}
sl@0
  1123
		
sl@0
  1124
		//updaterotation should not fail after this point (no cleanup)
sl@0
  1125
			
sl@0
  1126
        //update the last set config with the new rotation change so we don't incorrectly
sl@0
  1127
        //change the layer extents
sl@0
  1128
        if (iDisplayControl)
sl@0
  1129
            {
sl@0
  1130
            TDisplayConfiguration config;
sl@0
  1131
            config.SetRotation(static_cast<TDisplayConfiguration::TRotation>(newRotation));           
sl@0
  1132
            iConfigChangeNotifier->UpdateLastSetConfiguration(config);
sl@0
  1133
            }		
sl@0
  1134
		
sl@0
  1135
		TWservCrEvent crEvent(TWservCrEvent::EDeviceOrientationChanged,iScreenNumber,&gcOrientation);
sl@0
  1136
		TWindowServerEvent::NotifyDrawer(crEvent);
sl@0
  1137
		
sl@0
  1138
		if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
sl@0
  1139
			{
sl@0
  1140
			iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
sl@0
  1141
			}
sl@0
  1142
		
sl@0
  1143
		}
sl@0
  1144
	
sl@0
  1145
	iRootWindow->AdjustCoordsDueToRotation();
sl@0
  1146
	if (rotating)
sl@0
  1147
		{
sl@0
  1148
		if(BlankScreenOnRotation())
sl@0
  1149
			{
sl@0
  1150
			iRootWindow->ClearDisplay();
sl@0
  1151
			}
sl@0
  1152
		
sl@0
  1153
		CWsTop::ClearAllRedrawStores();	
sl@0
  1154
		DiscardAllSchedules();
sl@0
  1155
		iRootWindow->InvalidateWholeScreen();
sl@0
  1156
		}
sl@0
  1157
	return ETrue;
sl@0
  1158
	}
sl@0
  1159
sl@0
  1160
TPoint CScreen::PhysicalToLogical(TPoint aPhysicalPt)
sl@0
  1161
	{
sl@0
  1162
	const TSizeMode& mode=ScreenSizeModeData();
sl@0
  1163
	TPoint logicalPt;
sl@0
  1164
	if(!iDisplayMapping)
sl@0
  1165
		{
sl@0
  1166
		//old behaviour
sl@0
  1167
		logicalPt=aPhysicalPt-mode.iOrigin;
sl@0
  1168
		if (mode.iScreenScale.iWidth!=1)
sl@0
  1169
			logicalPt.iX=(logicalPt.iX>=0 ? logicalPt.iX/mode.iScreenScale.iWidth : (logicalPt.iX-(mode.iScreenScale.iWidth-1))/mode.iScreenScale.iWidth);
sl@0
  1170
		if (mode.iScreenScale.iHeight!=1)
sl@0
  1171
			logicalPt.iY=(logicalPt.iY>=0 ? logicalPt.iY/mode.iScreenScale.iHeight : (logicalPt.iY-(mode.iScreenScale.iHeight-1))/mode.iScreenScale.iHeight);
sl@0
  1172
		}
sl@0
  1173
	else
sl@0
  1174
		{
sl@0
  1175
		//rect with dummy size for coordinates mapping purpose
sl@0
  1176
		TRect rectInComp(aPhysicalPt, TSize(1,1));
sl@0
  1177
		TRect rectInApp;
sl@0
  1178
		iDisplayMapping->MapCoordinates(ECompositionSpace, rectInComp, EApplicationSpace,rectInApp);
sl@0
  1179
		logicalPt = rectInApp.iTl;
sl@0
  1180
		}
sl@0
  1181
	
sl@0
  1182
	return logicalPt;
sl@0
  1183
	}
sl@0
  1184
sl@0
  1185
void CScreen::LoadScreenSizesL(TSize aScreenSize)
sl@0
  1186
	{
sl@0
  1187
	_LIT(KWSERVNumScrSizeMode, "NUMSCREENMODES");
sl@0
  1188
	TBool allowScrGap=WsIniFile->FindVar(iScreenNumber, KWSERVNumScrSizeMode, iNumScreenSizeModes);
sl@0
  1189
	iModes=new(ELeave) RPointerArray<TInternalSizeMode>(1);
sl@0
  1190
	WS_ASSERT_DEBUG(!allowScrGap || (allowScrGap && iNumScreenSizeModes>0), EWsPanicInvalidScreenSizeMode);
sl@0
  1191
	TInt screenNum=0;
sl@0
  1192
	FOREVER
sl@0
  1193
		{
sl@0
  1194
		++screenNum;
sl@0
  1195
		TBuf<32> varNameWidth;
sl@0
  1196
		TBuf<32> varNameHeight;
sl@0
  1197
		_LIT(KWSERVScreenWidthPattern,"SCR_WIDTH%d");
sl@0
  1198
		varNameWidth.Format(KWSERVScreenWidthPattern,screenNum);
sl@0
  1199
		_LIT(KWSERVScreenHeightPattern,"SCR_HEIGHT%d");
sl@0
  1200
		varNameHeight.Format(KWSERVScreenHeightPattern,screenNum);
sl@0
  1201
		TSize screenSize;
sl@0
  1202
		if (!WsIniFile->FindVar(iScreenNumber, varNameWidth, screenSize.iWidth) ||
sl@0
  1203
			!WsIniFile->FindVar(iScreenNumber, varNameHeight, screenSize.iHeight))
sl@0
  1204
			{
sl@0
  1205
			if (allowScrGap && screenNum<=iNumScreenSizeModes)
sl@0
  1206
				{
sl@0
  1207
				iModes->AppendL(NULL);
sl@0
  1208
				continue;
sl@0
  1209
				}
sl@0
  1210
			else
sl@0
  1211
				break;
sl@0
  1212
			}
sl@0
  1213
		TInt flags=0;
sl@0
  1214
		if (screenSize.iWidth==0 && screenSize.iHeight==0)
sl@0
  1215
			{
sl@0
  1216
			screenSize=aScreenSize;
sl@0
  1217
			flags|=EHalDefault;
sl@0
  1218
			}
sl@0
  1219
		if (screenSize.iWidth==-1 && screenSize.iHeight==-1)
sl@0
  1220
			{
sl@0
  1221
			screenSize=aScreenSize;
sl@0
  1222
			flags|=EDynamic;
sl@0
  1223
			iFlags|=EHasDynamicSizeModes;
sl@0
  1224
			}
sl@0
  1225
		TInternalSizeMode* newSizeMode=new(ELeave) TInternalSizeMode(screenSize);
sl@0
  1226
		newSizeMode->iFlags|=flags;
sl@0
  1227
		CleanupStack::PushL(newSizeMode);
sl@0
  1228
		iModes->AppendL(newSizeMode);
sl@0
  1229
		CleanupStack::Pop(newSizeMode);
sl@0
  1230
		++iNumSupportedScreenSizeModes;
sl@0
  1231
		}
sl@0
  1232
	// If sparse index is enabled and no screen size mode defined, all iModes entries will be NULL
sl@0
  1233
	// Otherwise iModes will be empty
sl@0
  1234
	if (iModes->Count()==0 || iNumSupportedScreenSizeModes==0)
sl@0
  1235
		{
sl@0
  1236
		TInternalSizeMode* defaultSizeMode=new(ELeave) TInternalSizeMode(aScreenSize);
sl@0
  1237
		defaultSizeMode->iFlags|=EHalDefault;
sl@0
  1238
		if (iModes->Count()>0)
sl@0
  1239
			(*iModes)[0]=defaultSizeMode;
sl@0
  1240
		else
sl@0
  1241
			{
sl@0
  1242
			CleanupStack::PushL(defaultSizeMode);
sl@0
  1243
			iModes->AppendL(defaultSizeMode);
sl@0
  1244
			CleanupStack::Pop(defaultSizeMode);
sl@0
  1245
			}
sl@0
  1246
		++iNumSupportedScreenSizeModes;
sl@0
  1247
		}
sl@0
  1248
	if (!allowScrGap)
sl@0
  1249
		iNumScreenSizeModes=iNumSupportedScreenSizeModes;
sl@0
  1250
	}
sl@0
  1251
sl@0
  1252
void CScreen::LoadScreenRotationProperties(TInternalSizeMode& aMode, const TInt aModeIndex)
sl@0
  1253
	{
sl@0
  1254
	TBuf<32> varRotation;
sl@0
  1255
	_LIT(KWSERVScreenRotationPattern,"SCR_ROTATION%d");
sl@0
  1256
	varRotation.Format(KWSERVScreenRotationPattern,aModeIndex+1);		
sl@0
  1257
	TInt rotation=CFbsBitGc::EGraphicsOrientationNormal;
sl@0
  1258
	TUint allRotations=0;
sl@0
  1259
	TPtrC rotList(NULL,0);
sl@0
  1260
	if (WsIniFile->FindVar( iScreenNumber, varRotation,rotList))
sl@0
  1261
		{
sl@0
  1262
		TLex lex(rotList);
sl@0
  1263
		TBool foundOne=EFalse;
sl@0
  1264
		TInt rot;
sl@0
  1265
sl@0
  1266
		while (!lex.Eos())
sl@0
  1267
			{
sl@0
  1268
			if (!FindNextValue(lex, rot))
sl@0
  1269
				{
sl@0
  1270
				break;
sl@0
  1271
				}
sl@0
  1272
			if (rot<0 || rot>360)
sl@0
  1273
				{
sl@0
  1274
				continue;
sl@0
  1275
				}
sl@0
  1276
			if (rot>4)
sl@0
  1277
				{
sl@0
  1278
				rot/=90;
sl@0
  1279
				}
sl@0
  1280
			if (!foundOne)
sl@0
  1281
				{
sl@0
  1282
				rotation=rot;
sl@0
  1283
				foundOne=ETrue;
sl@0
  1284
				}
sl@0
  1285
			if (rot<=CFbsBitGc::EGraphicsOrientationRotated270)
sl@0
  1286
				{
sl@0
  1287
				allRotations|=ROTATION_TO_FLAG(rot);
sl@0
  1288
				}
sl@0
  1289
			}
sl@0
  1290
		}
sl@0
  1291
	if (allRotations==0)
sl@0
  1292
		allRotations=ROTATION_TO_FLAG(rotation);
sl@0
  1293
	const TInt KAllRotationsMask = 0xF;	//Used to keep the old behaviour
sl@0
  1294
	WS_ASSERT_ALWAYS((ROTATION_TO_FLAG(rotation)&KAllRotationsMask)>0, EWsPanicFailedToInitialise);
sl@0
  1295
	aMode.iRotation=reinterpret_cast<CFbsBitGc::TGraphicsOrientation&>(rotation);
sl@0
  1296
	aMode.iAlternativeRotations=allRotations&KAllRotationsMask;
sl@0
  1297
	}
sl@0
  1298
sl@0
  1299
void CScreen::LoadScreenTwipsProperties(TInternalSizeMode& aMode, const TInt aModeIndex)
sl@0
  1300
	{
sl@0
  1301
	TBuf<32> varNameWidth;
sl@0
  1302
	_LIT(KWSERVScreenTwipWidthPattern,"SCR_TWIP_WIDTH%d");
sl@0
  1303
	varNameWidth.Format(KWSERVScreenTwipWidthPattern,aModeIndex+1);
sl@0
  1304
	TBuf<32> varNameHeight;
sl@0
  1305
	_LIT(KWSERVScreenTwipHeightPattern,"SCR_TWIP_HEIGHT%d");
sl@0
  1306
	varNameHeight.Format(KWSERVScreenTwipHeightPattern,aModeIndex+1);
sl@0
  1307
sl@0
  1308
	TSize twipsSize;
sl@0
  1309
	TBool widthFound = WsIniFile->FindVar(iScreenNumber,varNameWidth,twipsSize.iWidth);
sl@0
  1310
	TBool heightFound = WsIniFile->FindVar(iScreenNumber,varNameHeight,twipsSize.iHeight);	
sl@0
  1311
	
sl@0
  1312
	// if either of the width or height wsini reads has failed we need to generate default values
sl@0
  1313
	switch(aMode.iRotation)
sl@0
  1314
		{
sl@0
  1315
		// CFbsBitGc::TGraphicsOrientation
sl@0
  1316
		case CFbsBitGc::EGraphicsOrientationRotated90:	// deliberate drop-through
sl@0
  1317
		case CFbsBitGc::EGraphicsOrientationRotated270:
sl@0
  1318
			{
sl@0
  1319
			// CFbsScreenDevice knows nothing about rotation, so we can't use it's PixelsTo Twips methods
sl@0
  1320
			// So swap the axis here to use the correct twips per pixel ratio
sl@0
  1321
			if (!widthFound)
sl@0
  1322
				twipsSize.iWidth = DeviceMap().VerticalPixelsToTwips(aMode.iScreenSize.iWidth);
sl@0
  1323
			if (!heightFound)
sl@0
  1324
				twipsSize.iHeight = DeviceMap().HorizontalPixelsToTwips(aMode.iScreenSize.iHeight);	
sl@0
  1325
			break;
sl@0
  1326
			}
sl@0
  1327
		case CFbsBitGc::EGraphicsOrientationNormal:		// deliberate drop-through
sl@0
  1328
		case CFbsBitGc::EGraphicsOrientationRotated180:
sl@0
  1329
			if (!widthFound)
sl@0
  1330
				twipsSize.iWidth = DeviceMap().HorizontalPixelsToTwips(aMode.iScreenSize.iWidth);
sl@0
  1331
			if (!heightFound)
sl@0
  1332
				twipsSize.iHeight = DeviceMap().VerticalPixelsToTwips(aMode.iScreenSize.iHeight);
sl@0
  1333
			break;
sl@0
  1334
		default:
sl@0
  1335
			RDebug::Print(_L("** CScreen::LoadScreenTwipsProperties Panic"));
sl@0
  1336
			WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
sl@0
  1337
			break;			
sl@0
  1338
		}
sl@0
  1339
	if (widthFound&&heightFound)
sl@0
  1340
		{
sl@0
  1341
		aMode.iFlags|=this->ETwipsSpecified;
sl@0
  1342
		}
sl@0
  1343
	aMode.iScreenTwipsSize=twipsSize;
sl@0
  1344
	}
sl@0
  1345
sl@0
  1346
sl@0
  1347
void CScreen::LoadScreenSizeProperties(TDisplayMode aDefaultDisplayMode)
sl@0
  1348
	{
sl@0
  1349
	for(TInt sizeLoop=0;sizeLoop<iModes->Count();sizeLoop++)
sl@0
  1350
		{
sl@0
  1351
		TInternalSizeMode* modePtr=(*iModes)[sizeLoop];
sl@0
  1352
		if (!modePtr)
sl@0
  1353
			continue;
sl@0
  1354
		TInternalSizeMode& mode=*modePtr;
sl@0
  1355
		TBuf<32> varDisplayMode;
sl@0
  1356
		_LIT(KWSERVScreenDisplayModePattern,"SCR_WINDOWMODE%d");
sl@0
  1357
		
sl@0
  1358
		varDisplayMode.Format(KWSERVScreenDisplayModePattern,sizeLoop+1);
sl@0
  1359
		mode.iScreenScale.iWidth=1;
sl@0
  1360
		mode.iScreenScale.iHeight=1;
sl@0
  1361
sl@0
  1362
		TBuf<32> varLeft;
sl@0
  1363
		TBuf<32> varTop;
sl@0
  1364
		_LIT(KWSERVScreenLeftPattern,"SCR_LEFT%d");
sl@0
  1365
		_LIT(KWSERVScreenTopPattern,"SCR_TOP%d");
sl@0
  1366
		varLeft.Format(KWSERVScreenLeftPattern,sizeLoop+1);
sl@0
  1367
		varTop.Format(KWSERVScreenTopPattern,sizeLoop+1);
sl@0
  1368
		if (!WsIniFile->FindVar( iScreenNumber, varLeft,mode.iOrigin.iX))
sl@0
  1369
			mode.iOrigin.iX=0;
sl@0
  1370
		if (!WsIniFile->FindVar( iScreenNumber, varTop,mode.iOrigin.iY))
sl@0
  1371
			mode.iOrigin.iY=0;
sl@0
  1372
			
sl@0
  1373
		TPtrC displayModeName(NULL,0);
sl@0
  1374
		mode.iDefaultDisplayMode = aDefaultDisplayMode;
sl@0
  1375
		// must know rotation before parsing twips
sl@0
  1376
		LoadScreenRotationProperties(mode, sizeLoop);
sl@0
  1377
		LoadScreenTwipsProperties(mode, sizeLoop);
sl@0
  1378
		
sl@0
  1379
		
sl@0
  1380
		if(mode.iScreenSize.iWidth == mode.iScreenSize.iHeight && mode.iAlternativeRotations == allRotationsMask)
sl@0
  1381
			{
sl@0
  1382
			//square appmode with all four rotations allowed must have square twipsize
sl@0
  1383
			if((mode.iFlags&ETwipsSpecified) && mode.iScreenTwipsSize.iWidth != mode.iScreenTwipsSize.iHeight)
sl@0
  1384
				{
sl@0
  1385
				RDebug::Print(_L("**Panic: Square appmode with all four rotations must have square twip size"));
sl@0
  1386
				WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
sl@0
  1387
				}
sl@0
  1388
			//square appmode with all four rotations allowed must have square offset
sl@0
  1389
			if(mode.iOrigin.iX != mode.iOrigin.iY)
sl@0
  1390
				{
sl@0
  1391
				RDebug::Print(_L("**Panic: Square appmode with all four rotations must have square offset"));
sl@0
  1392
				WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
sl@0
  1393
				}
sl@0
  1394
			}
sl@0
  1395
		else
sl@0
  1396
			{
sl@0
  1397
			//Everything else is treated as rectangle appmode. Square appmode not supporting all 4 rotations is considered
sl@0
  1398
			//as rectangle appmode as well. Rectangle appmode suports 2 rotations at most (0 and 180, or 90 and 270)
sl@0
  1399
			//first rotation of the appmode is taken to apply the corresponding rotation mask
sl@0
  1400
			if(!((mode.iAlternativeRotations&KRotation0_180Mask) == mode.iAlternativeRotations
sl@0
  1401
					|| (mode.iAlternativeRotations&KRotation90_270Mask) == mode.iAlternativeRotations))
sl@0
  1402
				{
sl@0
  1403
				RDebug::Print(_L("**Panic_DEBUG: non square appmode can only define (0,180) or (90,270) rotations"));
sl@0
  1404
				WS_PANIC_DEBUG(EWsPanicFailedToInitialise);
sl@0
  1405
				//in relase build, no panic, just correct the rotations set
sl@0
  1406
sl@0
  1407
				}
sl@0
  1408
			//correct the rotations set
sl@0
  1409
			mode.iAlternativeRotations &= ((ROTATION_TO_FLAG(mode.iRotation) & KRotation0_180Mask)? KRotation0_180Mask:
sl@0
  1410
												KRotation90_270Mask);
sl@0
  1411
			}
sl@0
  1412
			
sl@0
  1413
		}
sl@0
  1414
//
sl@0
  1415
	TInt intForFindVar=0;
sl@0
  1416
	_LIT(KWSERVIniFileVarSizeMode,"SIZE_MODE");
sl@0
  1417
	WsIniFile->FindVar( iScreenNumber, KWSERVIniFileVarSizeMode,intForFindVar);
sl@0
  1418
	iSizeEnforcementMode=(TScreenModeEnforcement)intForFindVar;
sl@0
  1419
	}
sl@0
  1420
sl@0
  1421
void CScreen::SetDigitiserAreas(const TSize& aUiSize)
sl@0
  1422
	{  //aUiSize should be the unrotated current ui size
sl@0
  1423
	//SetDigitiserAreas needs revisiting if/when we support dynamic resolutions on a screen 
sl@0
  1424
    //with touch input. It is not known how digitiser coordinates will be represented if the 
sl@0
  1425
    //physical display resolution changes. Currently digitisers are only supported on screen 0, 
sl@0
  1426
    //and dynamic resolution only applies to higher screen numbers on real hardware.
sl@0
  1427
	for(TInt sizeLoop=0;sizeLoop<iModes->Count();sizeLoop++)
sl@0
  1428
		{
sl@0
  1429
		TInternalSizeMode* modePtr=(*iModes)[sizeLoop];
sl@0
  1430
		if (!modePtr)
sl@0
  1431
			continue;
sl@0
  1432
		TInternalSizeMode& mode=*modePtr;
sl@0
  1433
		if(mode.iFlags & EClientDefinedDigitiserArea)
sl@0
  1434
			{
sl@0
  1435
			//if it's client set, keep it unchanged.
sl@0
  1436
			continue;
sl@0
  1437
			}
sl@0
  1438
		switch (mode.iRotation)
sl@0
  1439
			{
sl@0
  1440
			case CFbsBitGc::EGraphicsOrientationNormal:
sl@0
  1441
				mode.iPointerCursorArea=iDigitiserArea;
sl@0
  1442
				continue;
sl@0
  1443
			case CFbsBitGc::EGraphicsOrientationRotated90:
sl@0
  1444
				mode.iPointerCursorArea.SetRect(iDigitiserArea.iTl.iY,aUiSize.iWidth-iDigitiserArea.iBr.iX,
sl@0
  1445
																iDigitiserArea.iBr.iY,aUiSize.iWidth-iDigitiserArea.iTl.iX);
sl@0
  1446
				break;
sl@0
  1447
			case CFbsBitGc::EGraphicsOrientationRotated180:
sl@0
  1448
				mode.iPointerCursorArea.SetRect(-(iDigitiserArea.iBr-aUiSize),-(iDigitiserArea.iTl-aUiSize));
sl@0
  1449
				break;
sl@0
  1450
			case CFbsBitGc::EGraphicsOrientationRotated270:
sl@0
  1451
				mode.iPointerCursorArea.SetRect(aUiSize.iHeight-iDigitiserArea.iBr.iY,iDigitiserArea.iTl.iX,
sl@0
  1452
				        aUiSize.iHeight-iDigitiserArea.iTl.iY,iDigitiserArea.iBr.iX);
sl@0
  1453
				break;
sl@0
  1454
			default:
sl@0
  1455
			    WS_PANIC_ALWAYS(EWsPanicInvalidRotation);
sl@0
  1456
			}
sl@0
  1457
		}
sl@0
  1458
	}
sl@0
  1459
sl@0
  1460
void CScreen::GetScreenSizeAndRotation(TPixelsTwipsAndRotation &aSar, TInt aScreenMode)
sl@0
  1461
	{
sl@0
  1462
	TSizeMode& mode=*(*iModes)[aScreenMode];
sl@0
  1463
	aSar.iRotation=mode.iRotation;
sl@0
  1464
	aSar.iPixelSize=mode.iScreenSize;
sl@0
  1465
	aSar.iTwipsSize=mode.iScreenTwipsSize;
sl@0
  1466
	if (aSar.iTwipsSize.iWidth==0)
sl@0
  1467
		{
sl@0
  1468
		aSar.iTwipsSize.iWidth  = iDeviceMap->HorizontalPixelsToTwips(aSar.iPixelSize.iWidth);
sl@0
  1469
		aSar.iTwipsSize.iHeight = iDeviceMap->VerticalPixelsToTwips(aSar.iPixelSize.iHeight);
sl@0
  1470
		}
sl@0
  1471
	}
sl@0
  1472
sl@0
  1473
void CScreen::GetScreenSizeAndRotation(TPixelsAndRotation &aSar, TInt aScreenMode)
sl@0
  1474
	{
sl@0
  1475
	TSizeMode& mode=*(*iModes)[aScreenMode];
sl@0
  1476
	aSar.iRotation=mode.iRotation;
sl@0
  1477
	aSar.iPixelSize=mode.iScreenSize;
sl@0
  1478
	}
sl@0
  1479
sl@0
  1480
TBool CScreen::SetScreenModeEnforcement(TInt aMode)
sl@0
  1481
	{
sl@0
  1482
	if (aMode<0 || aMode>ESizeEnforcementPixelsTwipsAndRotation)
sl@0
  1483
		return EFalse;
sl@0
  1484
	TScreenModeEnforcement newMode=(TScreenModeEnforcement)aMode;
sl@0
  1485
	if (newMode!=iSizeEnforcementMode)
sl@0
  1486
		{
sl@0
  1487
		iSizeEnforcementMode=newMode;
sl@0
  1488
		CWsWindowGroup::SetScreenDeviceValidStates(this);
sl@0
  1489
		ResetFocus(NULL);
sl@0
  1490
		}
sl@0
  1491
	return ETrue;
sl@0
  1492
	}
sl@0
  1493
sl@0
  1494
void CScreen::IncContrast()
sl@0
  1495
	{
sl@0
  1496
	TInt contrast;
sl@0
  1497
	if (iMaxContrast<0)			//If failed to get it sofar get it again
sl@0
  1498
		TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Get(iScreenNumber,HALData::EDisplayContrastMax,iMaxContrast));
sl@0
  1499
	if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Get(iScreenNumber,HALData::EDisplayContrast,contrast)))
sl@0
  1500
		return;
sl@0
  1501
	if (contrast==iMaxContrast)
sl@0
  1502
		contrast=-1;
sl@0
  1503
	TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Set(iScreenNumber,HALData::EDisplayContrast,++contrast));
sl@0
  1504
	}
sl@0
  1505
sl@0
  1506
void CScreen::DecContrast()
sl@0
  1507
	{
sl@0
  1508
	TInt contrast;
sl@0
  1509
	if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Get(iScreenNumber,HALData::EDisplayContrast,contrast)))
sl@0
  1510
		return;
sl@0
  1511
	if (contrast==0)
sl@0
  1512
		{
sl@0
  1513
		if (iMaxContrast<0 && TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast,
sl@0
  1514
															HAL::Get(iScreenNumber,HALData::EDisplayContrastMax,iMaxContrast)))
sl@0
  1515
			return;
sl@0
  1516
		contrast=iMaxContrast+1;
sl@0
  1517
		}
sl@0
  1518
	TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Set(iScreenNumber,HALData::EDisplayContrast,--contrast));
sl@0
  1519
	}
sl@0
  1520
sl@0
  1521
void CScreen::IncBrightness()
sl@0
  1522
	{
sl@0
  1523
	TInt brightness;
sl@0
  1524
	if (iMaxBrightness<0)			//If failed to get it sofar get it again
sl@0
  1525
		TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(iScreenNumber,HALData::EDisplayBrightnessMax,iMaxBrightness));
sl@0
  1526
	if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(iScreenNumber,HALData::EDisplayBrightness,brightness)))
sl@0
  1527
		return;
sl@0
  1528
	if (brightness==iMaxBrightness)
sl@0
  1529
		brightness=-1;
sl@0
  1530
	TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(iScreenNumber,HALData::EDisplayBrightness,++brightness));
sl@0
  1531
	}
sl@0
  1532
sl@0
  1533
void CScreen::DecBrightness()
sl@0
  1534
	{
sl@0
  1535
	TInt brightness;
sl@0
  1536
	if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(iScreenNumber,HALData::EDisplayBrightness,brightness)))
sl@0
  1537
		return;
sl@0
  1538
	if (brightness==0)
sl@0
  1539
		{
sl@0
  1540
		if (iMaxBrightness<0 && TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight,
sl@0
  1541
													HAL::Get(iScreenNumber,HALData::EDisplayBrightnessMax,iMaxBrightness)))
sl@0
  1542
			return;
sl@0
  1543
		brightness=iMaxBrightness+1;
sl@0
  1544
		}
sl@0
  1545
	TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(iScreenNumber,HALData::EDisplayBrightness,--brightness));
sl@0
  1546
	}
sl@0
  1547
TInt CScreen::GetScreenSizeModeList(RArray<TInt>& aList) const
sl@0
  1548
	{
sl@0
  1549
	aList.Reset();
sl@0
  1550
	TInt numModes=iNumScreenSizeModes;
sl@0
  1551
	TInt err=aList.Reserve(numModes);
sl@0
  1552
	if (err!=KErrNone)
sl@0
  1553
		return err;
sl@0
  1554
	TInt index;
sl@0
  1555
	for (index=0; index<numModes; ++index)
sl@0
  1556
		{
sl@0
  1557
		TSizeMode* modePtr=(*iModes)[index];
sl@0
  1558
		if (modePtr)
sl@0
  1559
			aList.Append(index);	//Can't fail due to reserve
sl@0
  1560
		}
sl@0
  1561
	TInt count=aList.Count();
sl@0
  1562
	return count;
sl@0
  1563
	}
sl@0
  1564
sl@0
  1565
TInt CScreen::GetScreenSizeModeListL()
sl@0
  1566
	{
sl@0
  1567
	RArray<TInt> list;
sl@0
  1568
	CleanupClosePushL(list);
sl@0
  1569
	TInt count=GetScreenSizeModeList(list);
sl@0
  1570
		User::LeaveIfError(count);
sl@0
  1571
	CWsClient::ReplyBuf(&list[0], count*sizeof(TInt));
sl@0
  1572
	CleanupStack::PopAndDestroy(&list);
sl@0
  1573
	return count;
sl@0
  1574
	}
sl@0
  1575
sl@0
  1576
void CScreen::SetInitialScreenSizeModeAndRotation()
sl@0
  1577
	{	//Set first app mode that supports current supported rotations if available
sl@0
  1578
	TInt index;
sl@0
  1579
	TInt firstMode = -1;
sl@0
  1580
	TInt bestMode = -1;
sl@0
  1581
	// Since all screen rotation modes are supported.
sl@0
  1582
	TBitFlags32 rotationFlags = CFbsBitGc::EGraphicsOrientationNormal|CFbsBitGc::EGraphicsOrientationRotated90|CFbsBitGc::EGraphicsOrientationRotated180|CFbsBitGc::EGraphicsOrientationRotated270;
sl@0
  1583
	for (index=0; index<iModes->Count(); ++index)
sl@0
  1584
		{
sl@0
  1585
		TSizeMode* modePtr=(*iModes)[index];
sl@0
  1586
		if (modePtr)
sl@0
  1587
			{
sl@0
  1588
			if (firstMode == -1)
sl@0
  1589
				{
sl@0
  1590
				firstMode = index;
sl@0
  1591
				}
sl@0
  1592
			if (rotationFlags.IsSet((TInt)modePtr->iRotation))
sl@0
  1593
				{
sl@0
  1594
				bestMode = index;
sl@0
  1595
				break;
sl@0
  1596
				}
sl@0
  1597
			}
sl@0
  1598
		}
sl@0
  1599
sl@0
  1600
	if (bestMode != -1)	//found a mode that supports current supported rotations
sl@0
  1601
		{
sl@0
  1602
		iScreenSizeMode = bestMode;
sl@0
  1603
		}
sl@0
  1604
	else
sl@0
  1605
		{
sl@0
  1606
		if (firstMode != -1)	//could only find a mode that doesnt support current supported rotations
sl@0
  1607
			{
sl@0
  1608
			iScreenSizeMode = firstMode;
sl@0
  1609
			}
sl@0
  1610
		else
sl@0
  1611
			{
sl@0
  1612
			return;	//couldn't find a mode at all
sl@0
  1613
			}
sl@0
  1614
		}
sl@0
  1615
	if(iDisplayPolicy)
sl@0
  1616
		{
sl@0
  1617
		iDisplayPolicy->SetLastAppMode(iScreenSizeMode);
sl@0
  1618
		}
sl@0
  1619
sl@0
  1620
    SetDigitiserAreas(iScreenDevice->SizeInPixels()); //unrotated size in pixels
sl@0
  1621
    
sl@0
  1622
    // Here we are mixing CFbsBitGc::TGraphicsOrientation with MWsScene::TSceneRotation
sl@0
  1623
    // As they both have same values it is fine for now
sl@0
  1624
    iScene->SetSceneRotation(GcToScreen(ScreenSizeModeData().iRotation)); //set rotation
sl@0
  1625
	}
sl@0
  1626
sl@0
  1627
sl@0
  1628
TDisplayMode CScreen::FirstDefaultDisplayMode() const
sl@0
  1629
	{
sl@0
  1630
	TInt mode=-1;
sl@0
  1631
	while ((*iModes)[++mode]==NULL)
sl@0
  1632
		{
sl@0
  1633
		WS_ASSERT_DEBUG(mode<iModes->Count()-1,EWsPanicInvalidScreenSizeMode);
sl@0
  1634
		}
sl@0
  1635
	return((*iModes)[mode]->iDefaultDisplayMode);
sl@0
  1636
	}
sl@0
  1637
sl@0
  1638
void CScreen::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
sl@0
  1639
	{
sl@0
  1640
	iRedraw->AddRedrawRegion(aRegion, aSchedule, aDepth);
sl@0
  1641
	}
sl@0
  1642
sl@0
  1643
void CScreen::ScheduleRender(const TTimeIntervalMicroSeconds& aFromNow)
sl@0
  1644
	{
sl@0
  1645
	iRedraw->ScheduleRender(aFromNow);
sl@0
  1646
	}
sl@0
  1647
sl@0
  1648
void CScreen::DoRedrawNow()
sl@0
  1649
	{
sl@0
  1650
	iRedraw->DoRedrawNow();
sl@0
  1651
	}
sl@0
  1652
sl@0
  1653
// See CScreenRedraw::IsUpdatePending() for important notes on usage.
sl@0
  1654
void CScreen::RedrawNowIfPending()
sl@0
  1655
	{
sl@0
  1656
	if(iRedraw->IsUpdatePending())
sl@0
  1657
		DoRedrawNow();
sl@0
  1658
	}
sl@0
  1659
sl@0
  1660
TBool CScreen::IsQuickFadeScheduled( CWsWindow* aWin ) const
sl@0
  1661
	{
sl@0
  1662
	return iRedraw->IsQuickFadeScheduled( aWin );
sl@0
  1663
	}
sl@0
  1664
sl@0
  1665
void CScreen::RemoveFromQuickFadeList( CWsWindow* aWin )
sl@0
  1666
	{
sl@0
  1667
	iRedraw->RemoveFromQuickFadeList( aWin );
sl@0
  1668
	}
sl@0
  1669
sl@0
  1670
void CScreen::AcceptFadeRequest( CWsWindow* aWin, TBool aIsFaded )
sl@0
  1671
	{
sl@0
  1672
	iRedraw->AcceptFadeRequest( aWin, aIsFaded );
sl@0
  1673
	}
sl@0
  1674
sl@0
  1675
// implementing MWsScreen
sl@0
  1676
sl@0
  1677
const TTime& CScreen::Now() const
sl@0
  1678
	{
sl@0
  1679
	return iRedraw->Now();
sl@0
  1680
	}
sl@0
  1681
sl@0
  1682
void CScreen::ScheduleAnimation(TAnimType aType, const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow,const TTimeIntervalMicroSeconds& aFreq,const TTimeIntervalMicroSeconds& aStop, CWsWindow* aWindow)
sl@0
  1683
	{
sl@0
  1684
	iRedraw->ScheduleAnimation(aType, aRect,aFromNow,aFreq,aStop, aWindow);
sl@0
  1685
	}
sl@0
  1686
sl@0
  1687
TBool CScreen::IsScheduled(TAnimType aType, const TRect& aRect, CWsWindow* aWindow) const
sl@0
  1688
	{
sl@0
  1689
	return iRedraw->IsScheduled(aType, aRect, aWindow);
sl@0
  1690
	}
sl@0
  1691
sl@0
  1692
void CScreen::OnAnimation(TRequestStatus* aFinished)
sl@0
  1693
	{
sl@0
  1694
	iRedraw->OnAnimation(aFinished);
sl@0
  1695
	}
sl@0
  1696
	
sl@0
  1697
void CScreen::Redraw()
sl@0
  1698
	{
sl@0
  1699
	STACK_REGION bounds;
sl@0
  1700
	bounds.AddRect(DrawableArea());
sl@0
  1701
	AddRedrawRegion(bounds);
sl@0
  1702
	bounds.Close();
sl@0
  1703
	}
sl@0
  1704
sl@0
  1705
TBool CScreen::RedrawInvalid(const TArray<TGraphicDrawerId>& aInvalid)
sl@0
  1706
	{
sl@0
  1707
	TBool wasDirty = EFalse;
sl@0
  1708
	STACK_REGION bounds;
sl@0
  1709
	bounds.AddRect(DrawableArea());
sl@0
  1710
	STACK_REGION dirty;
sl@0
  1711
	TWalkWindowTreeCalcInvalidGraphics calc(&bounds,dirty,aInvalid);
sl@0
  1712
	if(calc.CreateSubRegion())
sl@0
  1713
		{
sl@0
  1714
		calc.CalcInvalid(*this);
sl@0
  1715
		if(dirty.CheckError() || dirty.Count())
sl@0
  1716
			{
sl@0
  1717
			Redraw();
sl@0
  1718
			wasDirty = ETrue;
sl@0
  1719
			}
sl@0
  1720
		calc.DestroyRegions();
sl@0
  1721
		}
sl@0
  1722
	dirty.Close();
sl@0
  1723
	bounds.Close();
sl@0
  1724
	return wasDirty;
sl@0
  1725
	}
sl@0
  1726
sl@0
  1727
/**
sl@0
  1728
 Overidding MWsObjectProvider
sl@0
  1729
*/
sl@0
  1730
TAny* CScreen::ResolveObjectInterface(TUint aTypeId)
sl@0
  1731
	{
sl@0
  1732
	TAny* interface = NULL;
sl@0
  1733
sl@0
  1734
	switch (aTypeId)
sl@0
  1735
		{
sl@0
  1736
		case MWsWindow::EWsObjectInterfaceId:
sl@0
  1737
			interface = static_cast<MWsWindow*>(RootWindow());
sl@0
  1738
			break;
sl@0
  1739
		case MWsScreenConfigList::EWsObjectInterfaceId:
sl@0
  1740
			interface = static_cast<MWsScreenConfigList*>(this);
sl@0
  1741
			break;
sl@0
  1742
		case MWsScreenConfig::EWsObjectInterfaceId:
sl@0
  1743
			interface = static_cast<MWsScreenConfig*>(this);
sl@0
  1744
			break;
sl@0
  1745
		case MWsWindowTree::EWsObjectInterfaceId:
sl@0
  1746
			interface = static_cast<MWsWindowTree*>(this);
sl@0
  1747
		}
sl@0
  1748
sl@0
  1749
	if (!interface)
sl@0
  1750
		interface = iRedraw->ResolveObjectInterface(aTypeId);
sl@0
  1751
sl@0
  1752
	return interface;
sl@0
  1753
	}
sl@0
  1754
sl@0
  1755
void CScreen::SendTree() const
sl@0
  1756
	{
sl@0
  1757
	if(!iWindowTreeObserver)
sl@0
  1758
		return;
sl@0
  1759
	
sl@0
  1760
	TWalkWindowTreeSendState wtw(*iWindowTreeObserver);
sl@0
  1761
	RootWindow()->WalkWindowTreeBackToFront(wtw, EVisitParentNodesFirst);
sl@0
  1762
sl@0
  1763
	//Standard text cursors
sl@0
  1764
	RWsTextCursor* cursor = CWsTop::CurrentTextCursor();
sl@0
  1765
	if(cursor)
sl@0
  1766
		cursor->SendState(*iWindowTreeObserver);
sl@0
  1767
	
sl@0
  1768
	//Floating Sprites	
sl@0
  1769
	SpriteManager()->SendState(*iWindowTreeObserver);
sl@0
  1770
	
sl@0
  1771
	//Window Group Chains
sl@0
  1772
	for(CWsWindowGroup *group=RootWindow()->Child(); group!=NULL; group=group->NextSibling())
sl@0
  1773
		{
sl@0
  1774
		group->SendStateWindowGroupChain(*iWindowTreeObserver);
sl@0
  1775
		}
sl@0
  1776
	}
sl@0
  1777
sl@0
  1778
TDisplayMode CScreen::DisplayMode() const
sl@0
  1779
	{
sl@0
  1780
	return iScreenDevice->DisplayMode();
sl@0
  1781
	}
sl@0
  1782
sl@0
  1783
TSize CScreen::SizeInPixels() const
sl@0
  1784
	{
sl@0
  1785
	return iScreenDevice->SizeInPixels();
sl@0
  1786
	}
sl@0
  1787
sl@0
  1788
TSize CScreen::SizeInTwips() const
sl@0
  1789
	{
sl@0
  1790
	return iScreenDevice->SizeInTwips();
sl@0
  1791
	}
sl@0
  1792
sl@0
  1793
void CScreen::DiscardAllSchedules()
sl@0
  1794
	{
sl@0
  1795
	iRedraw->DiscardAllSchedules();
sl@0
  1796
	}
sl@0
  1797
sl@0
  1798
void CScreen::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
sl@0
  1799
	{
sl@0
  1800
	iRedraw->ScheduleRegionUpdate(aDefinitelyDirty);
sl@0
  1801
	}
sl@0
  1802
sl@0
  1803
TBool CScreen::IsDSAClientWindow( const CWsClientWindow* aWin ) const
sl@0
  1804
	{
sl@0
  1805
	TBool res = EFalse; 
sl@0
  1806
	if ( ! iDirects.IsEmpty() )
sl@0
  1807
		{
sl@0
  1808
		TSglQueIter<CWsDirectScreenAccess> iter( (TSglQueBase&)iDirects );
sl@0
  1809
		iter.SetToFirst();
sl@0
  1810
		CWsDirectScreenAccess* dsa;
sl@0
  1811
		while ( (dsa = iter++) != NULL && !res )
sl@0
  1812
			{
sl@0
  1813
			res =  (dsa->ClientWindow() == aWin) && (dsa->IsVisible());
sl@0
  1814
			}
sl@0
  1815
		}
sl@0
  1816
	return res;	
sl@0
  1817
	}
sl@0
  1818
sl@0
  1819
/**
sl@0
  1820
Update the UI element composition method based on whether
sl@0
  1821
there are any externals surfaces present and the current display mode. 
sl@0
  1822
If the method changes, recomposition is triggered.
sl@0
  1823
*/
sl@0
  1824
void CScreen::UpdateCompositionMode()
sl@0
  1825
	{
sl@0
  1826
	// do nothing
sl@0
  1827
	}
sl@0
  1828
sl@0
  1829
void CScreen::ElementAdded()
sl@0
  1830
	{
sl@0
  1831
	UpdateCompositionMode();
sl@0
  1832
	}
sl@0
  1833
sl@0
  1834
void CScreen::ElementRemoved()
sl@0
  1835
	{
sl@0
  1836
	UpdateCompositionMode();
sl@0
  1837
	}
sl@0
  1838
sl@0
  1839
CRegisteredSurfaceMap* CScreen::SurfaceMap()
sl@0
  1840
	{
sl@0
  1841
	return iSurfaceMap;
sl@0
  1842
	}
sl@0
  1843
sl@0
  1844
void CScreen::InitializeSceneL()
sl@0
  1845
	{
sl@0
  1846
	// Ensure the surface is not valid to start with.
sl@0
  1847
	iDsaSurface = TSurfaceId::CreateNullId();
sl@0
  1848
	iWindowElementSet = CWindowElementSet::NewL(*iScene);
sl@0
  1849
	}
sl@0
  1850
sl@0
  1851
MWsElement* CScreen::CreateUiElementL(const TRect& aExtent)
sl@0
  1852
	{
sl@0
  1853
	MWsElement* pElement = iScene->CreateSceneElementL();
sl@0
  1854
	
sl@0
  1855
	TUint32 flags = 0;
sl@0
  1856
	pElement->GetRenderStageFlags(flags);
sl@0
  1857
	flags |= MWsElement::EElementIsIndirectlyRenderedUserInterface;
sl@0
  1858
	pElement->SetRenderStageFlags(flags);
sl@0
  1859
	    
sl@0
  1860
    iScene->InsertSceneElement(pElement, NULL);
sl@0
  1861
sl@0
  1862
	pElement->SetDestinationRectangle(aExtent);
sl@0
  1863
	pElement->SetSourceRectangle(aExtent);	//initial guess... updated by PositionUiElements
sl@0
  1864
sl@0
  1865
	return pElement;
sl@0
  1866
	}
sl@0
  1867
sl@0
  1868
void CScreen::InitializeUiElementsL()
sl@0
  1869
	{
sl@0
  1870
	const TRect screenRect(iScreenDevice->SizeInPixels());
sl@0
  1871
	MWsElement* pElement = CreateUiElementL(screenRect);
sl@0
  1872
	
sl@0
  1873
	if(HasAlpha())
sl@0
  1874
		{
sl@0
  1875
		TUint32 flags = 0;
sl@0
  1876
		pElement->GetTargetRendererFlags(flags);
sl@0
  1877
		flags |= MWsElement::EElementTransparencySource;
sl@0
  1878
		pElement->SetTargetRendererFlags(flags);
sl@0
  1879
		}
sl@0
  1880
	}
sl@0
  1881
sl@0
  1882
TInt CScreen::InitializeDsaSurface()
sl@0
  1883
	{
sl@0
  1884
	WS_ASSERT_DEBUG(iDsaSurface.IsNull(),EWsPanicInvalidOperation);
sl@0
  1885
	iDsaDevice->GetSurface(iDsaSurface);
sl@0
  1886
	// Currently Surface Manager does not recognise DSA surface IDs originating
sl@0
  1887
	// from the Screen Driver.  This causes it to fail to register such
sl@0
  1888
	// surfaces.  OpenWF should be amended to properly register DSA surfaces.
sl@0
  1889
	iScene->RegisterSurface(iDsaSurface);
sl@0
  1890
sl@0
  1891
	return KErrNone;
sl@0
  1892
	}
sl@0
  1893
sl@0
  1894
TSize CScreen::DSASizeInPixels() const
sl@0
  1895
	{
sl@0
  1896
	if(iDsaDevice)
sl@0
  1897
		{
sl@0
  1898
		return iDsaDevice->SizeInPixels();
sl@0
  1899
		}
sl@0
  1900
	else
sl@0
  1901
		{
sl@0
  1902
		return TSize(0,0);
sl@0
  1903
		}
sl@0
  1904
	}
sl@0
  1905
sl@0
  1906
MWsTextCursor* CScreen::RenderStageTextCursor() const
sl@0
  1907
	{
sl@0
  1908
	return iRedraw->RenderStageTextCursor();
sl@0
  1909
	}
sl@0
  1910
sl@0
  1911
void CScreen::ClearDsaSurface(const TRect& area, const TRgb& color)
sl@0
  1912
	{
sl@0
  1913
	WS_ASSERT_DEBUG(iDsaGc, EWsPanicInvalidOperation);
sl@0
  1914
	iDsaGc->SetBrushStyle(CFbsBitGc::ESolidBrush);
sl@0
  1915
	iDsaGc->SetPenStyle(CFbsBitGc::ENullPen);
sl@0
  1916
	iDsaGc->SetBrushColor(color);
sl@0
  1917
	iDsaGc->DrawRect(area);
sl@0
  1918
	iDsaDevice->Update();
sl@0
  1919
	}
sl@0
  1920
sl@0
  1921
void CScreen::ReleaseDsaScreenDevice()
sl@0
  1922
	{
sl@0
  1923
	//This function checks if any of the DSA currently active on the screen is actually used to draw
sl@0
  1924
	//If not it unregister the DSA surface and destroys the iDsaDevice.
sl@0
  1925
	//This function should be called only by a drawing DSA so a surface should be in place.
sl@0
  1926
	iNumberDrawingDsa--;
sl@0
  1927
	if(iNumberDrawingDsa == 0)
sl@0
  1928
		{
sl@0
  1929
		WS_ASSERT_DEBUG(!iDsaSurface.IsNull(),EWsPanicInvalidOperation);
sl@0
  1930
		delete iDsaGc;
sl@0
  1931
		iDsaGc = NULL;
sl@0
  1932
		// Currently Surface Manager does not recognise DSA surface IDs originating
sl@0
  1933
		// from the Screen Driver.  This causes it to fail to unregister such
sl@0
  1934
		// surfaces.  OpenWF should be amended to properly register DSA surfaces.
sl@0
  1935
		iScene->UnregisterSurface(iDsaSurface);
sl@0
  1936
sl@0
  1937
		delete iDsaDevice;
sl@0
  1938
		iDsaDevice = NULL;
sl@0
  1939
		//the old surface Id is now meaningless
sl@0
  1940
		iDsaSurface = TSurfaceId::CreateNullId();
sl@0
  1941
		}
sl@0
  1942
	}
sl@0
  1943
sl@0
  1944
TInt CScreen::SetConfiguration(const TDisplayConfiguration& aConfigInput)
sl@0
  1945
	{
sl@0
  1946
	TInt reply = KErrNone;
sl@0
  1947
	if(iDisplayControl)
sl@0
  1948
		{
sl@0
  1949
		TDisplayConfiguration config(aConfigInput);
sl@0
  1950
		TRect sizeModePosition;
sl@0
  1951
		if (iDisplayPolicy)
sl@0
  1952
			{	//validate config and update to a valid hardware config
sl@0
  1953
			reply = iDisplayPolicy->GetSizeModeConfiguration(iScreenSizeMode,config,sizeModePosition);
sl@0
  1954
			if (reply >= KErrNone)
sl@0
  1955
				{//set appmode in policy
sl@0
  1956
				if (iDisplayMapping)
sl@0
  1957
					{
sl@0
  1958
					iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
sl@0
  1959
					}
sl@0
  1960
				}
sl@0
  1961
			}
sl@0
  1962
		else
sl@0
  1963
			{	//exessive strategy: limit rotation agains curr app mode.
sl@0
  1964
				//really we want the system to accept the rotation change regardless of the app mode.
sl@0
  1965
			TDisplayConfiguration::TRotation newRot;
sl@0
  1966
			if (aConfigInput.GetRotation(newRot))
sl@0
  1967
				{	//This should cast between rotation enumertaions "properly"
sl@0
  1968
				if (!(iModes[0][iScreenSizeMode]->iAlternativeRotations&(1<<newRot)))
sl@0
  1969
					{
sl@0
  1970
					reply=KErrArgument;
sl@0
  1971
					}
sl@0
  1972
				}
sl@0
  1973
			}
sl@0
  1974
		if (reply < KErrNone)
sl@0
  1975
			{
sl@0
  1976
			return reply;
sl@0
  1977
			}
sl@0
  1978
		MWsScene::TSceneRotation oldRotation;
sl@0
  1979
		oldRotation = iScene->SceneRotation();
sl@0
  1980
		TSize newUiSize;
sl@0
  1981
		config.GetResolution(newUiSize);
sl@0
  1982
		TDisplayConfiguration oldConfig;
sl@0
  1983
		iDisplayControl->GetConfiguration(oldConfig);
sl@0
  1984
		if(iFlags&EHasDynamicSizeModes)
sl@0
  1985
			{
sl@0
  1986
			reply = iFallbackMap->Resize(newUiSize);
sl@0
  1987
			}
sl@0
  1988
		if (reply >= KErrNone)
sl@0
  1989
			{
sl@0
  1990
			reply=iDisplayControl->SetConfiguration(config);
sl@0
  1991
			}	
sl@0
  1992
		if (reply==KErrNone)
sl@0
  1993
			{
sl@0
  1994
			TSize oldConfigRes;
sl@0
  1995
			oldConfig.GetResolution(oldConfigRes);
sl@0
  1996
            TDisplayConfiguration newConfig;
sl@0
  1997
			if (oldConfigRes.iWidth == 0 || oldConfigRes.iHeight == 0)
sl@0
  1998
				{
sl@0
  1999
				iDisplayControl->GetConfiguration(newConfig);
sl@0
  2000
				RecalculateModeTwips(&newConfig);	//needs res and twips information
sl@0
  2001
				}
sl@0
  2002
			UpdateDynamicScreenModes();
sl@0
  2003
			
sl@0
  2004
			//update the last set config in the config change notifier to 
sl@0
  2005
			//prevent SetConfiguration() from being called again!
sl@0
  2006
			newConfig.ClearAll();
sl@0
  2007
			iDisplayControl->GetConfiguration(newConfig);
sl@0
  2008
			iConfigChangeNotifier->UpdateLastSetConfiguration(newConfig); 			
sl@0
  2009
			
sl@0
  2010
			TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EScreenSizeModeAboutToChange, iScreenSizeMode));
sl@0
  2011
			// This will remove all the DSA elements from the scene
sl@0
  2012
			AbortAllDirectDrawing(RDirectScreenAccess::ETerminateRotation);
sl@0
  2013
			
sl@0
  2014
			//SetDigitiserAreas needs revisiting if/when we support dynamic resolutions
sl@0
  2015
			//on a screen with touch input.
sl@0
  2016
			//SetDigitiserAreas(newUiSize);
sl@0
  2017
			
sl@0
  2018
			//failure here should only be because of DSA orientation change failure, which shouldn't happen, either.
sl@0
  2019
			//Or there may be no change to do.
sl@0
  2020
			(void)UpdateOrientation(&oldRotation);
sl@0
  2021
			
sl@0
  2022
			iWindowElementSet->ResubmitAllElementExtents();
sl@0
  2023
			if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
sl@0
  2024
				{
sl@0
  2025
				iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
sl@0
  2026
				}
sl@0
  2027
			
sl@0
  2028
			iRootWindow->AdjustCoordsDueToRotation();
sl@0
  2029
sl@0
  2030
			//TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
sl@0
  2031
			if(BlankScreenOnRotation())
sl@0
  2032
				{
sl@0
  2033
				iRootWindow->ClearDisplay();
sl@0
  2034
				}
sl@0
  2035
sl@0
  2036
			CWsTop::ClearAllRedrawStores();
sl@0
  2037
			DiscardAllSchedules();
sl@0
  2038
			iRootWindow->InvalidateWholeScreen();
sl@0
  2039
			CWsWindowGroup::SetScreenDeviceValidStates(this);
sl@0
  2040
			TWindowServerEvent::SendScreenDeviceChangedEvents(this);
sl@0
  2041
			}
sl@0
  2042
		else
sl@0
  2043
			{
sl@0
  2044
			return reply;
sl@0
  2045
			}
sl@0
  2046
		}
sl@0
  2047
	else
sl@0
  2048
		{
sl@0
  2049
		reply = KErrNotSupported;
sl@0
  2050
		}
sl@0
  2051
	return reply;
sl@0
  2052
	}
sl@0
  2053
sl@0
  2054
/**
sl@0
  2055
 * Updates the screen device display properties. This is to ensure the screen device is 
sl@0
  2056
 * consistent with any configuration changes not made using CScreen::SetConfiguration.
sl@0
  2057
 * 
sl@0
  2058
 * @param aConfigInput a fully populated display configuration
sl@0
  2059
 **/
sl@0
  2060
TInt CScreen::UpdateConfiguration(const TDisplayConfiguration& aConfigInput)
sl@0
  2061
    {
sl@0
  2062
    TInt reply = KErrNone;
sl@0
  2063
    if(iDisplayControl)
sl@0
  2064
        {
sl@0
  2065
        TDisplayConfiguration config(aConfigInput);
sl@0
  2066
        TRect sizeModePosition;
sl@0
  2067
        if (iDisplayPolicy)
sl@0
  2068
            {   //validate config and update to a valid hardware config
sl@0
  2069
            reply = iDisplayPolicy->GetSizeModeConfiguration(iScreenSizeMode,config,sizeModePosition);
sl@0
  2070
            if (reply >= KErrNone)
sl@0
  2071
                {//set appmode in policy
sl@0
  2072
                if (iDisplayMapping)
sl@0
  2073
                    {
sl@0
  2074
                    iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
sl@0
  2075
                    }
sl@0
  2076
                }
sl@0
  2077
            }
sl@0
  2078
        else
sl@0
  2079
            {   //exessive strategy: limit rotation agains curr app mode.
sl@0
  2080
                //really we want the system to accept the rotation change regardless of the app mode.
sl@0
  2081
            TDisplayConfiguration::TRotation newRot;
sl@0
  2082
            if (aConfigInput.GetRotation(newRot))
sl@0
  2083
                {   //This should cast between rotation enumertaions "properly"
sl@0
  2084
                if (!(iModes[0][iScreenSizeMode]->iAlternativeRotations&(1<<newRot)))
sl@0
  2085
                    {
sl@0
  2086
                    reply=KErrArgument;
sl@0
  2087
                    }
sl@0
  2088
                }
sl@0
  2089
            }
sl@0
  2090
sl@0
  2091
        MWsScene::TSceneRotation oldRotation;
sl@0
  2092
        oldRotation = iScene->SceneRotation();
sl@0
  2093
        TSize newUiSize;
sl@0
  2094
        config.GetResolution(newUiSize);
sl@0
  2095
        if(iFlags&EHasDynamicSizeModes)
sl@0
  2096
            {
sl@0
  2097
            reply = iFallbackMap->Resize(newUiSize);
sl@0
  2098
            }
sl@0
  2099
sl@0
  2100
        RecalculateModeTwips(&config);   //needs res and twips information
sl@0
  2101
        UpdateDynamicScreenModes();        
sl@0
  2102
        
sl@0
  2103
        TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EScreenSizeModeAboutToChange, iScreenSizeMode));
sl@0
  2104
        // This will remove all the DSA elements from the scene
sl@0
  2105
        AbortAllDirectDrawing(RDirectScreenAccess::ETerminateRotation);
sl@0
  2106
        
sl@0
  2107
        //SetDigitiserAreas needs revisiting if/when we support dynamic resolutions
sl@0
  2108
        //on a screen with touch input.
sl@0
  2109
        //SetDigitiserAreas(newUiSize);
sl@0
  2110
        
sl@0
  2111
        //failure here should only be because of DSA orientation change failure, which shouldn't happen, either.
sl@0
  2112
        //Or there may be no change to do.
sl@0
  2113
        (void)UpdateOrientation(&oldRotation);
sl@0
  2114
        
sl@0
  2115
        iWindowElementSet->ResubmitAllElementExtents();
sl@0
  2116
        if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
sl@0
  2117
            {
sl@0
  2118
            iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
sl@0
  2119
            }
sl@0
  2120
        
sl@0
  2121
        iRootWindow->AdjustCoordsDueToRotation();
sl@0
  2122
sl@0
  2123
        //TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
sl@0
  2124
        if(BlankScreenOnRotation())
sl@0
  2125
            {
sl@0
  2126
            iRootWindow->ClearDisplay();
sl@0
  2127
            }
sl@0
  2128
sl@0
  2129
        CWsTop::ClearAllRedrawStores();
sl@0
  2130
        DiscardAllSchedules();
sl@0
  2131
        iRootWindow->InvalidateWholeScreen();
sl@0
  2132
        CWsWindowGroup::SetScreenDeviceValidStates(this);
sl@0
  2133
        TWindowServerEvent::SendScreenDeviceChangedEvents(this);
sl@0
  2134
        }
sl@0
  2135
    else
sl@0
  2136
        {
sl@0
  2137
        reply = KErrNotSupported;
sl@0
  2138
        }
sl@0
  2139
    return reply;  
sl@0
  2140
    }
sl@0
  2141
sl@0
  2142
void CScreen::UpdateDynamicScreenModes()
sl@0
  2143
	{
sl@0
  2144
	WS_ASSERT_DEBUG(iDisplayControl,EWsPanicNoDisplayControl);
sl@0
  2145
	TDisplayConfiguration newConfig;
sl@0
  2146
	iDisplayControl->GetConfiguration(newConfig);
sl@0
  2147
	TSize res;
sl@0
  2148
	TSize twips;
sl@0
  2149
	newConfig.GetResolution(res);
sl@0
  2150
	newConfig.GetResolutionTwips(twips);
sl@0
  2151
	for (TInt i=0; i<iModes->Count(); i++)
sl@0
  2152
		{
sl@0
  2153
		if ((*iModes)[i] && ((*iModes)[i]->iFlags & EDynamic))
sl@0
  2154
			{
sl@0
  2155
			(*iModes)[i]->iScreenSize = res;
sl@0
  2156
			(*iModes)[i]->iScreenTwipsSize = twips;
sl@0
  2157
			}
sl@0
  2158
		}
sl@0
  2159
	}
sl@0
  2160
sl@0
  2161
void CScreen::RecalculateModeTwips(const TDisplayConfiguration* aConfig)
sl@0
  2162
	{
sl@0
  2163
	TDisplayConfiguration config;
sl@0
  2164
	iDisplayControl->GetConfiguration(config);
sl@0
  2165
	TSize res;
sl@0
  2166
	TSize twips;
sl@0
  2167
	if (aConfig)	//called from SetConfiguration
sl@0
  2168
		{
sl@0
  2169
		aConfig->GetResolution(res);
sl@0
  2170
		if (res.iWidth == 0 || res.iHeight == 0)
sl@0
  2171
			{
sl@0
  2172
			return;
sl@0
  2173
			}
sl@0
  2174
		aConfig->GetResolutionTwips(twips);
sl@0
  2175
		}
sl@0
  2176
	else	//called from DisplayChangeNotifier during attach
sl@0
  2177
		{
sl@0
  2178
		config.GetResolution(res);
sl@0
  2179
		if ((res.iWidth == 0 || res.iHeight == 0) && !iDisplayPolicy)
sl@0
  2180
			{
sl@0
  2181
			return;
sl@0
  2182
			}
sl@0
  2183
		config.GetResolutionTwips(twips);
sl@0
  2184
		}
sl@0
  2185
	TInt err=KErrNone;
sl@0
  2186
	TInt flags=0;
sl@0
  2187
	for (TInt ii=0; ii<iModes->Count(); ii++)
sl@0
  2188
		{	//for every mode
sl@0
  2189
		TRAP(err, flags = ModePtrL(ii)->iFlags);
sl@0
  2190
		if (err != KErrNone || flags&(ETwipsSpecified|EDynamic))
sl@0
  2191
			{	//continue if mode doesnt exist,twips specified or dynamic mode specified
sl@0
  2192
			continue;
sl@0
  2193
			}
sl@0
  2194
sl@0
  2195
		if (iDisplayPolicy)
sl@0
  2196
			{	//get ideal config for app mode from policy
sl@0
  2197
			TRect modePosition;
sl@0
  2198
			config.ClearAll();
sl@0
  2199
			TInt err = iDisplayPolicy->GetSizeModeConfiguration(ii,config,modePosition);
sl@0
  2200
			if (err != KErrNone)
sl@0
  2201
				{	//nothing we can do, the twips will not be calculated correctly
sl@0
  2202
				continue;
sl@0
  2203
				}
sl@0
  2204
			config.GetResolution(res);
sl@0
  2205
			config.GetResolutionTwips(twips);
sl@0
  2206
			}
sl@0
  2207
		TSizeMode* modePtr=(*iModes)[ii];
sl@0
  2208
		modePtr->iScreenTwipsSize.iWidth = (twips.iWidth * modePtr->iScreenSize.iWidth)/
sl@0
  2209
				res.iWidth;
sl@0
  2210
		modePtr->iScreenTwipsSize.iHeight = (twips.iHeight * modePtr->iScreenSize.iHeight)/
sl@0
  2211
				res.iHeight;
sl@0
  2212
		}
sl@0
  2213
	
sl@0
  2214
	}
sl@0
  2215
sl@0
  2216
TInt CScreen::AddNotificationClient(CWsClient *aClient)
sl@0
  2217
	{
sl@0
  2218
	TInt err = iWsClientList.InsertInAddressOrder(aClient);
sl@0
  2219
	if(!(err == KErrNone || err == KErrAlreadyExists))
sl@0
  2220
		{
sl@0
  2221
		return err;
sl@0
  2222
		}
sl@0
  2223
	return KErrNone;
sl@0
  2224
	}
sl@0
  2225
void CScreen::RemoveNotificationClient(CWsClient *aClient)
sl@0
  2226
	{
sl@0
  2227
	TInt index = iWsClientList.FindInAddressOrder(aClient);
sl@0
  2228
	if(index != KErrNotFound)
sl@0
  2229
		{
sl@0
  2230
		iWsClientList.Remove(index);
sl@0
  2231
		}
sl@0
  2232
	}
sl@0
  2233
TInt CScreen::GetNotificationClients(RPointerArray<CWsClient>& aClientsArray)
sl@0
  2234
	{
sl@0
  2235
	TInt err = aClientsArray.Reserve(iWsClientList.Count());
sl@0
  2236
	if(err != KErrNone)
sl@0
  2237
		return err;
sl@0
  2238
	
sl@0
  2239
	for(TInt i = 0; i < iWsClientList.Count(); i++)
sl@0
  2240
		{
sl@0
  2241
		aClientsArray.Append(iWsClientList[i]);
sl@0
  2242
		}
sl@0
  2243
	return KErrNone;
sl@0
  2244
	}
sl@0
  2245
sl@0
  2246
TInt CScreen::FindNotificationClient (CWsClient *aClient)
sl@0
  2247
	{
sl@0
  2248
	return iWsClientList.FindInAddressOrder(aClient);
sl@0
  2249
	}
sl@0
  2250
sl@0
  2251
// implementing MWsScreenConfig... this might be better as RS interface
sl@0
  2252
TSize CScreen::ScreenModeSizeInPixels() const
sl@0
  2253
	{
sl@0
  2254
	return (*iModes)[iScreenSizeMode]->iScreenSize;
sl@0
  2255
	}
sl@0
  2256
TInt CScreen::Stride() const
sl@0
  2257
	{
sl@0
  2258
	return 0;
sl@0
  2259
	}
sl@0
  2260
TInt CScreen::SizeMode() const
sl@0
  2261
	{
sl@0
  2262
	return iScreenSizeMode;
sl@0
  2263
sl@0
  2264
	}
sl@0
  2265
TSize CScreen::ScalingFactor() const
sl@0
  2266
	{
sl@0
  2267
	return (*iModes)[iScreenSizeMode]->iScreenScale;
sl@0
  2268
	}
sl@0
  2269
TPoint CScreen::Origin() const
sl@0
  2270
	{
sl@0
  2271
	return (*iModes)[iScreenSizeMode]->iOrigin;
sl@0
  2272
	}
sl@0
  2273
TPoint CScreen::ScaledOrigin() const
sl@0
  2274
	{
sl@0
  2275
	return (*iModes)[iScreenSizeMode]->ScaledOrigin();
sl@0
  2276
	}
sl@0
  2277
sl@0
  2278
const CScreen::TInternalSizeMode* CScreen::ModePtrL(TInt aIndex) const
sl@0
  2279
	{
sl@0
  2280
	if (aIndex>=iModes->Count() || aIndex<0)
sl@0
  2281
		{
sl@0
  2282
		User::Leave(KErrArgument);
sl@0
  2283
		}
sl@0
  2284
	if (iModes==NULL)
sl@0
  2285
		{
sl@0
  2286
		User::Leave(KErrNotReady);
sl@0
  2287
		}
sl@0
  2288
	TInternalSizeMode* modePtr=(*iModes)[aIndex];
sl@0
  2289
	if (modePtr==NULL)
sl@0
  2290
		{
sl@0
  2291
		User::Leave(KErrArgument);
sl@0
  2292
		}
sl@0
  2293
	return modePtr;
sl@0
  2294
	}
sl@0
  2295
sl@0
  2296
TDisplayMode CScreen::DisplayModeL(TInt aIndex) const
sl@0
  2297
	{
sl@0
  2298
	return ModePtrL(aIndex)->iDefaultDisplayMode;
sl@0
  2299
	}
sl@0
  2300
TSize CScreen::ScreenModeSizeInPixelsL(TInt aIndex) const
sl@0
  2301
	{
sl@0
  2302
	return ModePtrL(aIndex)->iScreenSize;
sl@0
  2303
	}
sl@0
  2304
TSize CScreen::ScreenModeSizeInTwipsL(TInt aIndex) const 
sl@0
  2305
	{
sl@0
  2306
	return ModePtrL(aIndex)->iScreenTwipsSize;
sl@0
  2307
	}
sl@0
  2308
sl@0
  2309
CFbsBitGc::TGraphicsOrientation CScreen::OrientationL(TInt aIndex) const
sl@0
  2310
	{
sl@0
  2311
	return ModePtrL(aIndex)->iRotation;
sl@0
  2312
	}
sl@0
  2313
TInt CScreen::AvailableOrientationsL(TInt aIndex) const
sl@0
  2314
	{
sl@0
  2315
	return ModePtrL(aIndex)->iAlternativeRotations;
sl@0
  2316
	}
sl@0
  2317
TSize CScreen::ScalingFactorL(TInt aIndex) const
sl@0
  2318
	{
sl@0
  2319
	return ModePtrL(aIndex)->iScreenScale;
sl@0
  2320
	}
sl@0
  2321
TPoint CScreen::OriginL(TInt aIndex) const
sl@0
  2322
	{
sl@0
  2323
	return ModePtrL(aIndex)->iOrigin;
sl@0
  2324
	}
sl@0
  2325
TPoint CScreen::ScaledOriginL(TInt aIndex) const
sl@0
  2326
	{
sl@0
  2327
	return ModePtrL(aIndex)->ScaledOrigin();
sl@0
  2328
	}
sl@0
  2329
TInt CScreen::ModeFlagsL(TInt aIndex) const
sl@0
  2330
	{
sl@0
  2331
	return ModePtrL(aIndex)->iFlags;
sl@0
  2332
	}
sl@0
  2333
void CScreen::SetCurrentScreenModeAttributes(const TSizeMode &aModeData)
sl@0
  2334
	{	
sl@0
  2335
	TSizeMode* modeToOverwrite=(*iModes)[iScreenSizeMode];
sl@0
  2336
	*modeToOverwrite=aModeData;
sl@0
  2337
	}
sl@0
  2338
sl@0
  2339
void CScreen::ScheduleWindow(CWsWindow* aWindow)
sl@0
  2340
	{
sl@0
  2341
	iRedraw->ScheduleWindow(aWindow);
sl@0
  2342
	}
sl@0
  2343
sl@0
  2344
void CScreen::RemoveFromScheduledList(CWsWindow* aWindow)
sl@0
  2345
	{
sl@0
  2346
	iRedraw->RemoveFromScheduledList(aWindow);
sl@0
  2347
	}
sl@0
  2348
sl@0
  2349
void CScreen::RemoveFromTimedDrawList(CWsWindow* aWindow)
sl@0
  2350
	{
sl@0
  2351
	iRedraw->RemoveFromTimedDrawList(aWindow);
sl@0
  2352
	}
sl@0
  2353
sl@0
  2354
void CScreen::SetupVisibleRegionTracking(CWsWindow& aWindow, TBool aRegister) const
sl@0
  2355
	{
sl@0
  2356
	if(ChangeTracking() && iWindowVisibilityNotifier)
sl@0
  2357
		{
sl@0
  2358
		if(aRegister)
sl@0
  2359
			{
sl@0
  2360
			iWindowVisibilityNotifier->RegisterWindow(aWindow);
sl@0
  2361
			}
sl@0
  2362
		else
sl@0
  2363
			{
sl@0
  2364
			iWindowVisibilityNotifier->UnregisterWindow(aWindow);
sl@0
  2365
			}
sl@0
  2366
		}
sl@0
  2367
	}
sl@0
  2368
sl@0
  2369
TBool CScreen::IsAnimating() const
sl@0
  2370
    {
sl@0
  2371
    return iRedraw->IsAnimating();
sl@0
  2372
    }