First public contribution.
1 // Copyright (c) 2006-2010 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
19 #include <graphics/wsscreendevice.h>
20 #include <graphics/wsscene.h>
21 #include <graphics/wselement.h>
25 #include "walkwindowtree.h"
27 #include "windowgroup.h"
30 #include "windowelementset.h"
31 #include "registeredsurfacemap.h"
33 #include "ScreenRedraw.h"
34 #include "wspluginmanager.h"
35 #include "devicemap.h"
36 #include <graphics/wsdisplaymapping.h>
37 #if defined(__WINS__) && defined(_DEBUG)
38 #include "../../debuglog/osbwin.h"
40 #include <graphics/wsdisplaycontrol.h>
42 GLREF_D CDebugLogBase *wsDebugLog;
44 LOCAL_C inline MWsScene::TSceneRotation GcToScreen(CFbsBitGc::TGraphicsOrientation aGcOrientation)
46 MWsScene::TSceneRotation screenRotation = MWsScene::ESceneAntiClockwise0;
48 switch (aGcOrientation)
50 case CFbsBitGc::EGraphicsOrientationRotated90:
51 screenRotation = MWsScene::ESceneAntiClockwise90;
53 case CFbsBitGc::EGraphicsOrientationRotated180:
54 screenRotation = MWsScene::ESceneAntiClockwise180;
56 case CFbsBitGc::EGraphicsOrientationRotated270:
57 screenRotation = MWsScene::ESceneAntiClockwise270;
61 return screenRotation;
64 LOCAL_C inline TDeviceOrientation GcToDevice(CFbsBitGc::TGraphicsOrientation aGcOrientation)
66 // Each device orientation is defined to be the value where just the bit
67 // corresponding to the GDI orientation value is set.
68 return (TDeviceOrientation)(1 << aGcOrientation);
71 LOCAL_D TBool FindNextValue(TLex& aLex, TInt& aValue) // assumes the list cannot contain *negative* integers
75 const TUint character=aLex.Peek();
76 if (Rng(TUint('0'), character, TUint('9')) || (character=='-'))
83 return (aLex.Val(aValue)==KErrNone);
86 CScreen::CFallbackMap * CScreen::CFallbackMap::NewL(CScreen* aScreen)
88 CFallbackMap * self = new (ELeave) CFallbackMap(aScreen);
89 CleanupStack::PushL(self);
91 CleanupStack::Pop(self);
95 CScreen::CFallbackMap::CFallbackMap(CScreen* aScreen) :
97 iRegion(TRect(TPoint(0,0), TSize(1,1))),
102 CScreen::CFallbackMap::~CFallbackMap()
107 void CScreen::CFallbackMap::ConstructL()
111 for (TInt num = 0; num < iScreen->NumScreenSizeModes(); ++num)
113 if (iScreen->IsValidScreenSizeMode(num))
115 const TSizeMode & mode = iScreen->ScreenSizeModeData(num);
116 TInt width = mode.iScreenSize.iWidth / 32;
117 if (mode.iScreenSize.iWidth & (32 - 1))
119 TInt size = width * mode.iScreenSize.iHeight;
125 iMap = new (ELeave) TInt [iMapSize];
128 void CScreen::CFallbackMap::Prepare()
130 const TSizeMode & mode = iScreen->ScreenSizeModeData();
131 Mem::FillZ(iMap, iMapSize * sizeof(TInt));
132 iCount = mode.iScreenSize.iHeight * mode.iScreenSize.iWidth;
133 WS_ASSERT_DEBUG(iRegion.Count() == 1, EWsPanicScreenFallback);
135 iRegion.AddRect(TRect(mode.iOrigin, mode.iScreenSize));
138 TBool CScreen::CFallbackMap::FillRegion(const TRegion& aRegion)
140 WS_ASSERT_DEBUG(!aRegion.CheckError(), EWsPanicScreenFallback);
141 if (aRegion.Count() > 20 || aRegion.CheckError())
146 const TRect * rect = aRegion.RectangleList();
147 for (TInt num = 0; num < aRegion.Count(); ++num)
149 hit = FillRect(*rect) || hit;
158 // x >> 5 is equivalent to x / 32
159 // 0x1F is the rounding error when dividing by 32
161 // The compiler might do all the optimizations for us - not checked.
162 TBool CScreen::CFallbackMap::FillRect(const TRect& aRect)
165 const TSizeMode & mode = iScreen->ScreenSizeModeData();
166 TRect scrrect(mode.iOrigin, mode.iScreenSize);
168 rect.Intersection(scrrect);
169 TInt rowWidthInInts = mode.iScreenSize.iWidth;
170 if (rowWidthInInts & 0x1F)
171 rowWidthInInts += 0x20;
172 rowWidthInInts >>= 5;
174 TInt colStartInInts = rect.iTl.iX >> 5;
175 TInt firstOffsetInBits = rect.iTl.iX & 0x1F;
177 for(TInt row = rect.iTl.iY; row < rect.iBr.iY; ++row)
179 TInt * map = iMap + row * rowWidthInInts + colStartInInts;
180 TInt offsetShift = 31 - firstOffsetInBits;
181 for (TInt col = rect.iTl.iX; col < rect.iBr.iX; ++col)
183 WS_ASSERT_DEBUG(map - iMap < iMapSize, EWsPanicScreenFallback);
184 if (!(*map & 1 << offsetShift))
190 (*map) |= (1 << offsetShift);
203 TInt CScreen::CFallbackMap::Count() const
208 const TRect * CScreen::CFallbackMap::Rect() const
210 return iRegion.RectangleList();
213 const RRegion * CScreen::CFallbackMap::Region() const
218 TInt CScreen::CFallbackMap::Resize(const TSize& aSize)
221 TInt width = (aSize.iWidth+31) >> 5; // divide by 32
222 TInt invertedWidth = (aSize.iHeight+31) >> 5; // divide by 32
224 TInt maxSize = Max(width * aSize.iHeight,invertedWidth * aSize.iWidth);
225 if (maxSize > iMapSize)
228 newMap = new TInt [maxSize];
246 CScreen::CScreen(): iDirects(_FOFF(CWsDirectScreenAccess,iLink)), iMaxContrast(-1), iMaxBrightness(-1)
247 , iDisplayChangeSpinner(0), iConfigChangeSpinner(0)
257 for(ii=iNumScreenSizeModes-1;ii>=0;--ii)
259 delete (*iModes)[ii];
265 iScreenDevice = NULL;
269 delete iSpriteManager;
270 delete iWindowElementSet;
271 if (!iDsaSurface.IsNull())
273 TInt err = iScene->UnregisterSurface(iDsaSurface);
274 WS_ASSERT_DEBUG(KErrNone == err, EWsPanicDirectScreenAccess);
279 #if defined(__WINS__) && defined(_DEBUG)
283 if(iDisplayChangeNotifier)
284 delete iDisplayChangeNotifier;
285 if(iConfigChangeNotifier)
286 delete iConfigChangeNotifier;
287 iWsClientList.Close();
290 void CScreen::ConstructL(const TRect& aDigitiserArea, TInt aScreenNumber)
292 iScreenNumber = aScreenNumber ;
296 _LIT(KWSERVInitScreen,"Initialising for Screen %d");
297 wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant, KWSERVInitScreen, aScreenNumber);
300 // create screen redraw with render stages
301 iRedraw = CScreenRedraw::NewL(*this);
302 iScreenDevice = iRedraw->ObjectInterface<MWsScreenDevice>();
303 WS_ASSERT_ALWAYS(iScreenDevice, EWsPanicScreenDeviceMissing);
304 iScene = iRedraw->ObjectInterface<MWsScene>();
305 WS_ASSERT_ALWAYS(iScene, EWsPanicSceneMissing);
307 iDeviceMap = CGraphicsDeviceMap::NewL(*iScreenDevice);
309 iDisplayControl = MWsScreen::ObjectInterface<MWsDisplayControl>();
310 iDisplayMapping = MWsScreen::ObjectInterface<MWsDisplayMapping>();
311 iDisplayPolicy = MWsScreen::ObjectInterface<MWsDisplayPolicy>();
314 // initialize screen size mode data
315 LoadScreenSizesL(iScreenDevice->SizeInPixels());
316 iFallbackMap = CFallbackMap::NewL(this);
318 LoadScreenSizeProperties(iScreenDevice->DisplayMode());
322 iDisplayPolicy->NewAppModesAvailable();
325 iDigitiserArea = aDigitiserArea;
326 SetInitialScreenSizeModeAndRotation(); //get the first/lowest valid mode. Also calls SetDigitiserAreas
328 ApplyRemainingWsiniSettingsL();
330 iRootWindow = new (ELeave) CWsRootWindow(NULL, this);
331 iRootWindow->ConstructL();
333 // Default fading parameters
337 iSpriteManager = CWsSpriteManager::NewL();
340 InitializeUiElementsL();
341 iSurfaceMap = new (ELeave) CRegisteredSurfaceMap(*iScene);
343 //if the interface for notification is available. start notification AO.
346 iDisplayChangeNotifier = CWsDisplayChangeNotifier::NewL(iDisplayControl, this);
347 iDisplayChangeNotifier->IssueNotificationRequest();
348 iConfigChangeNotifier = CWsConfigChangeNotifier::NewL(iDisplayControl, this);
349 iConfigChangeNotifier->IssueNotificationRequest();
351 doSetScreenMode(iScreenSizeMode,ETrue);
354 void CScreen::ApplyRemainingWsiniSettingsL()
356 #if defined(__WINS__) && defined(_DEBUG)
357 _LIT(KDebugOsb,"DEBUGOSB");
358 if(WsIniFile->FindVar(iScreenNumber, KDebugOsb))
360 _LIT(KDebugWinTitleFormat, "Screen %d, DSA surface");
362 title.Format(KDebugWinTitleFormat, iScreenNumber);
363 iDebugWin = CDebugOsbWin::NewL(title, iScreenDevice->SizeInPixels());
368 _LIT(KWSERVIniFileVarAutoClear,"AUTOCLEAR");
369 WsIniFile->FindVar(iScreenNumber,KWSERVIniFileVarAutoClear,autoClear);
375 _LIT(KBackLight,"BACKLIGHTCONTROL");
376 iBackLightFlag=WsIniFile->FindVar( iScreenNumber, KBackLight);
378 _LIT(KWSERVIniFileVarBlankScreen, "BLANKSCREENONROTATION");
379 if (WsIniFile->FindVar(iScreenNumber, KWSERVIniFileVarBlankScreen))
381 iFlags|=EBlankScreenOnRotation;
384 //Cache pointers to renderstage APIs required in CHANGETRACKING mode
385 iWindowTreeObserver = iRedraw->ObjectInterface<MWsWindowTreeObserver>();
386 iDrawAnnotationObserver = iRedraw->ObjectInterface<MWsDrawAnnotationObserver>();
387 iWindowVisibilityNotifier = iRedraw->ObjectInterface<MWsWindowVisibilityNotifier>();
388 if(iWindowVisibilityNotifier)
390 iWindowVisibilityNotifier->RegisterWindowVisibilityObserver(iRedraw);
392 if (WsIniFile->FindVar(iScreenNumber, KWSERVIniFileVarChangeTracking))
394 iFlags|=EChangeTracking;
398 TInt refreshRate = 1000000;
399 _LIT(KDebugBar, "DEBUGBAR");
400 if (WsIniFile->FindVar(KDebugBar, refreshRate))
402 if (refreshRate < 100000)
403 // coverity [dead_error_line]
405 iDebugBar = CDebugBar::NewL(this, refreshRate);
409 void CScreen::AcquireDsaScreenDeviceL()
411 //creates WSERV's DSA buffer handle
412 //registers the DSA surface into the scene accordingly to the value of aRegisterSurface
415 TDisplayMode screenMode = ENone;
416 screenMode = iScreenDevice->DisplayMode();
417 if(screenMode != ENone)
419 CreateDsaScreenDeviceIfSupportedL(screenMode);
421 iDsaDevice->SetAutoUpdate(EFalse);
422 iDsaDevice->SetDeviceOrientation(GcToDevice(ScreenSizeModeData().iRotation));
423 iDsaDevice->ChangeScreenDevice(NULL); //This is necessary to initialise the screen
424 // register DSA Surface
425 TInt err = InitializeDsaSurface();
426 // create a graphics context to clear the DSA surface
428 err = iDsaDevice->CreateContext(iDsaGc);
430 iDsaGc->Activate(iDsaDevice);
433 //Creation of the DSA surface failed
434 //Cleanup the DSA Surface ID
435 iDsaSurface = TSurfaceId::CreateNullId();
444 User::Leave(KErrNotSupported);
450 void CScreen::CreateDsaScreenDeviceIfSupportedL(TDisplayMode aScreenMode)
452 if(DoCreateDsaScreenDevice(aScreenMode))
454 // Try creating the screen device with all available display modes, going from best to worst
455 __ASSERT_COMPILE(EColorLast == 14); // if any display mode is added to TDisplayMode we must update the list below
456 // (the list below contains all enums in TDisplayMode except ENone, ERgb, EColorLast)
457 if(DoCreateDsaScreenDevice(EColor16MAP))
459 if(DoCreateDsaScreenDevice(EColor16MA))
461 if(DoCreateDsaScreenDevice(EColor16MU))
463 if(DoCreateDsaScreenDevice(EColor16M))
465 if(DoCreateDsaScreenDevice(EColor64K))
467 if(DoCreateDsaScreenDevice(EColor4K))
469 if(DoCreateDsaScreenDevice(EColor256))
471 if(DoCreateDsaScreenDevice(EColor16))
473 if(DoCreateDsaScreenDevice(EGray256))
475 if(DoCreateDsaScreenDevice(EGray16))
477 if(DoCreateDsaScreenDevice(EGray4))
479 if(DoCreateDsaScreenDevice(EGray2))
481 User::Leave(KErrNotSupported);
484 TBool CScreen::DoCreateDsaScreenDevice(TDisplayMode aScreenMode)
486 TRAPD(err, iDsaDevice = CFbsScreenDevice::NewL(iScreenNumber, aScreenMode));
489 TUint supportedDsaRotationModes = iDsaDevice->DeviceOrientationsAvailable();
490 MWsScene::TSceneRotation currenTSceneRotation = iScene->SceneRotation();
491 TBool doesDsaSupportThisMode = EFalse;
492 switch(currenTSceneRotation)
494 case MWsScene::ESceneAntiClockwise0:
495 if(supportedDsaRotationModes & EDeviceOrientationNormal)
497 doesDsaSupportThisMode = ETrue;
500 case MWsScene::ESceneAntiClockwise90:
501 if(supportedDsaRotationModes & EDeviceOrientation90CW)
503 doesDsaSupportThisMode = ETrue;
506 case MWsScene::ESceneAntiClockwise180:
507 if(supportedDsaRotationModes & EDeviceOrientation180)
509 doesDsaSupportThisMode = ETrue;
512 case MWsScene::ESceneAntiClockwise270:
513 if(supportedDsaRotationModes & EDeviceOrientation270CW)
515 doesDsaSupportThisMode = ETrue;
519 RDebug::Print(_L("** CScreen::DoCreateDsaScreenDevice Panic, non existing rotation mode"));
520 WS_PANIC_ALWAYS(EWsPanicInvalidOperation);
523 if(!doesDsaSupportThisMode)
527 RDebug::Print(_L("** Current Rotation Mode not supported by the DSA device"));
528 err = KErrNotSupported;
531 return (err == KErrNone);
534 void CScreen::AbortDSAs(RDirectScreenAccess::TTerminationReasons aReason, TSglQue<CWsDirectScreenAccess>& aDirects)
536 if (aDirects.IsEmpty())
540 CWsDirectScreenAccess* direct= NULL;
541 TSglQueIter<CWsDirectScreenAccess> iter(aDirects);
542 while ((direct=iter++)!=NULL)
545 direct->SignalAbort(aReason);
548 TRequestStatus timerStatus;
549 RTimer& timer=CWsTop::Timer();
552 TRequestStatus** cancelReqList = (TRequestStatus**) User::AllocZ(sizeof(TRequestStatus*) * (nofDSAs + 1));
553 if (NULL != cancelReqList)
556 timer.After(timerStatus, KDSAAbortingImmediateRespAwaitFrameMicrosec);
558 while ((direct=iter++)!=NULL)
560 WS_ASSERT_DEBUG((dsaNo<=(nofDSAs)),EWsPanicDirectScreenAccess);
561 cancelReqList[ dsaNo ] = &direct->AbortStatus();
564 cancelReqList[ 0 ] = &timerStatus;
566 //wait for response or timeout
567 User::WaitForNRequest(cancelReqList, nofDSAs + 1);
570 while ((direct=iter++)!=NULL)
572 if (direct->AbortStatus() != KRequestPending)
573 direct->CancelAbortObject(); // responded
578 if (timerStatus == KRequestPending)
581 User::WaitForRequest(timerStatus);
584 User::Free(cancelReqList);
589 while ((direct=iter++) != NULL)
591 TRequestStatus timerStatus;
592 RTimer& timer=CWsTop::Timer();
594 timer.After(timerStatus, KDSAAbortingImmediateRespAwaitFrameMicrosec);
596 //wait for response or timeout
597 User::WaitForRequest(direct->AbortStatus(), timerStatus);
599 if (direct->AbortStatus() != KRequestPending)
600 direct->CancelAbortObject(); //responded
602 direct->Abort(); //timed out
604 if (timerStatus == KRequestPending)
607 User::WaitForRequest(timerStatus);
613 void CScreen::AbortAllDirectDrawing(RDirectScreenAccess::TTerminationReasons aReason)
615 AbortDSAs(aReason,iDirects);
618 void CScreen::AddDirect(CWsDirectScreenAccess& aDirect)
620 TBool emptyBefore = iDirects.IsEmpty();
621 iDirects.AddLast(aDirect);
622 TBool emptyAfter = iDirects.IsEmpty();
623 if (emptyBefore && ! emptyAfter)
626 wsevent.SetType(EEventDirectScreenAccessBegin);
627 *(wsevent.Int()) = iScreenNumber;
628 TWindowServerEvent::PublishNotification(wsevent);
631 if (iDsaDrawState==EDsaDrawStateIdle && aDirect.IsVisible())
633 iDsaDrawState = EDsaDrawStateDrawing;
634 TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EDsaDrawingBegin, iScreenNumber));
638 void CScreen::RemoveDirect(CWsDirectScreenAccess& aDirect)
640 TBool emptyBefore = iDirects.IsEmpty();
641 iDirects.Remove(aDirect);
642 TBool emptyAfter = iDirects.IsEmpty();
643 if (emptyAfter && ! emptyBefore)
646 wsevent.SetType(EEventDirectScreenAccessEnd);
647 *(wsevent.Int()) = iScreenNumber;
648 TWindowServerEvent::PublishNotification(wsevent);
651 if (iDsaDrawState==EDsaDrawStateDrawing && aDirect.IsVisible() && !HasVisibleDirectOnQueue())
653 iDsaDrawState = EDsaDrawStateIdle;
654 TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EDsaDrawingEnd, iScreenNumber));
658 TBool CScreen::HasVisibleDirectOnQueue()
660 if (iDirects.IsEmpty())
663 TSglQueIter<CWsDirectScreenAccess> iter(iDirects);
664 CWsDirectScreenAccess* dsa;
665 while ((dsa=iter++)!=NULL)
667 if (dsa->IsVisible())
675 TBool CScreen::IsDirectOnQueue(const CWsDirectScreenAccess* aDirect)
677 TSglQueIter<CWsDirectScreenAccess> iter(iDirects);
678 CWsDirectScreenAccess* direct;
679 while ((direct=iter++)!=NULL)
688 void CScreen::KillForegroundSession()
692 _LIT(KWSERVKillWinGp,"Killing Session owning Window Group with Id=%d");
694 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,KWSERVKillWinGp,iCurrentFocus->Identifier());
695 iCurrentFocus->WsOwner()->SessionTerminate();
699 CWsWindowGroup* CScreen::FindNewFocus(CWsRootWindow* aRootWindow)
701 CWsWindowGroup* newFocus;
702 for(newFocus=aRootWindow->Child();newFocus && newFocus->CanReceiveFocus()==EFalse;newFocus=newFocus->NextSibling()) {}
707 void CScreen::ResetFocus(CWsWindowGroup* aClosingWindow)
709 CWsWindowGroup* oldFocus=iCurrentFocus;
710 CWsWindowGroup* newFocus=NULL;
711 CScreen* newFocusedScreen=NULL;
712 iCurrentFocus=FindNewFocus(iRootWindow);
713 TBool focusedScreen= EFalse;
714 /*Focus policy is specified in the wsini.ini file using the keyword 'MULTIFOCUSPOLICY'.
715 If the keyword is not specified, then the default policy is run.
717 if(!CWsTop::MultiFocusPolicy())
719 focusedScreen=(this==CWsTop::CurrentFocusScreen()); //check if this screen is the focus screen
720 if (!iCurrentFocus && focusedScreen)
722 /*If this screen is the focused screen but does not have a focusable window group, then search for the
723 next screen that has a focusable window group and set that screen as the focused screen.
725 CScreen* screen=NULL;
727 for (screenNo=0; screenNo<CWsTop::NumberOfScreens() && !newFocus; ++screenNo)
729 if (screenNo!=iScreenNumber)
731 screen=CWsTop::Screen(screenNo);
732 newFocus=FindNewFocus(screen->RootWindow());
736 newFocusedScreen=screen;
739 /*Scenario A: multi-focus policy
740 newFocusedScreen is NULL
741 focusedScreen is EFalse
742 CWsTop::MultiFocusPolicy() returns ETrue
743 Check if the new focusable window group is not the same, send focus lost message to window group
744 that has just lost focus and send focus gain message to window group that can receive focus.
745 Scenario B: single-focus policy (default)
746 CWsTop::MultiFocusPolicy() returns EFalse
747 Check if the new focusable window group is not the same or if there is a new focused screen, send focus lost
748 message to window group that has just lost focus and send focus gain message to window group that can receive focus.
750 if (iCurrentFocus!=oldFocus || newFocusedScreen)
752 if (oldFocus && (focusedScreen || CWsTop::MultiFocusPolicy()) && oldFocus!=aClosingWindow)
754 oldFocus->LostFocus();
756 if (newFocusedScreen)
758 CWsTop::SetCurrentFocusScreen(newFocusedScreen);
759 newFocus->ReceivedFocus();
761 else if (iCurrentFocus && (focusedScreen || CWsTop::MultiFocusPolicy()))
763 iCurrentFocus->ReceivedFocus();
765 TWsPointer::UpdatePointerCursor();
766 TWindowServerEvent::SendFocusChangedEvents();
768 TWindowServerEvent::SendGroupListChangedEvents();
771 void CScreen::RemoveFromDefaultOwningList(CWsWindowGroup *aDestroyedGroup)
773 for (CWsWindowGroup **group=&iDefaultOwningWindow;*group;group=(*group)->NextDefaultOwningWindowPtr())
775 if (*group==aDestroyedGroup)
777 *group=*aDestroyedGroup->NextDefaultOwningWindowPtr();
783 void CScreen::SetDefaultOwningWindow(CWsWindowGroup *aGroup)
785 RemoveFromDefaultOwningList(aGroup);
786 aGroup->SetNextDefaultOwningWindow(iDefaultOwningWindow);
787 iDefaultOwningWindow=aGroup;
790 void CScreen::GetScanLine(const TWsSdCmdGetScanLine *aGetScanLine)
792 TRgb buf[EGetScanLineBufLen];
793 TPtr8 des((TUint8 *)&buf[0],EGetScanLineBufLen*sizeof(TRgb));
794 TPoint pos(aGetScanLine->pos);
796 TInt len=(des.MaxLength()*EGetScanLineBufLen)/CFbsBitmap::ScanLineLength(EGetScanLineBufLen,aGetScanLine->dispMode);
797 if (aGetScanLine->len < 0 || (CFbsBitmap::ScanLineLength(aGetScanLine->len, aGetScanLine->dispMode) >
798 CWsClient::CurrentClient()->ClientMessage().GetDesMaxLength(1)))
800 CWsClient::PanicCurrentClient(EWservPanicInvalidParameter);
804 if ((aGetScanLine->len-read)<len)
805 len=aGetScanLine->len-read;
806 iScreenDevice->GetScanLine(des,pos,len,aGetScanLine->dispMode);
807 CWsClient::ReplyBuf(des);
809 if (read==aGetScanLine->len)
815 void CScreen::MaxNumColors(TInt& aColors,TInt& aGrays)
818 aColors=TDisplayModeUtils::NumDisplayModeColors(DisplayMode());
819 if (!TDisplayModeUtils::IsDisplayModeColor(DisplayMode()))
826 #define MODE_TO_FLAG(x) 1<<(x-1)
827 #define ROTATION_TO_FLAG(x) 1<<x
828 const TUint allRotationsMask = ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationNormal)
829 | ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated90)
830 | ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated180)
831 | ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated270);
832 const TUint KRotation0_180Mask = ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationNormal)
833 | ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated180);
834 const TUint KRotation90_270Mask = ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated90)
835 | ROTATION_TO_FLAG(CFbsBitGc::EGraphicsOrientationRotated270);
836 TInt CScreen::ColorModesFlag()
838 return MODE_TO_FLAG(DisplayMode());
841 void CScreen::UpdateDsa()
845 #if defined(__WINS__) && defined(_DEBUG)
847 iDebugWin->Refresh(iDsaDevice->SizeInPixels(), iDsaDevice->DisplayMode(), iDsaDevice->Bits());
850 iDsaDevice->Update();
854 const CGraphicsDeviceMap& CScreen::DeviceMap() const
859 const MWsScreenDevice& CScreen::ScreenDevice() const
861 return *iScreenDevice;
864 void CScreen::SetPointerCursorArea(TInt aMode,const TRect& aRect)
866 (*iModes)[aMode]->iPointerCursorArea=aRect;
867 (*iModes)[aMode]->iFlags |= EClientDefinedDigitiserArea;
868 TWsPointer::SetPointerCursorPos(TWsPointer::PointerCursorPos());
871 CFbsBitGc::TGraphicsOrientation CScreen::Orientation() const
873 WS_ASSERT_DEBUG(IsValidScreenSizeMode(iScreenSizeMode),EWsPanicInvalidScreenSizeMode);
874 return (*iModes)[iScreenSizeMode]->iRotation;
877 TRect CScreen::DrawableArea() const
879 TRect drawRect=iScreenDevice->SizeInPixels();
882 iDisplayMapping->MapCoordinates(EFullScreenSpace,drawRect,EApplicationSpace,drawRect);
887 TClientPanic CScreen::SetModeRotation(TInt aMode,CFbsBitGc::TGraphicsOrientation aRotation)
889 if (!IsValidScreenSizeMode(aMode))
890 return EWservPanicScreenModeNumber;
891 TSizeMode& mode=*(*iModes)[aMode];
892 if (!(ROTATION_TO_FLAG(aRotation)&mode.iAlternativeRotations))
893 return EWservPanicRotation;
894 CFbsBitGc::TGraphicsOrientation oldRotation=mode.iRotation;
895 mode.iRotation=aRotation;
896 CWsWindowGroup::NewOrientation(aMode,aRotation, iRootWindow);
897 if (aMode==ScreenSizeMode())
899 if(iDisplayPolicy && iDisplayControl && (mode.iAlternativeRotations == allRotationsMask))
901 //square mode supports all 4 rotations. We'd better do a complete Setconfiguration
902 //all parameters are recalculated for 90 degree rotation
903 //The most important one is the offset, since we need to re-center the appmode and it's a policy behaviour
904 TDisplayConfiguration config;
905 iDisplayControl->GetConfiguration(config);
907 config.SetRotation((TDisplayConfiguration::TRotation)aRotation);
908 SetConfiguration(config);
910 else if (!UpdateOrientation())
912 // Roll back earlier change
913 mode.iRotation=oldRotation;
914 CWsWindowGroup::NewOrientation(aMode, oldRotation, iRootWindow);
917 return EWservNoPanic;
920 void CScreen::CycleDisplaySize()
922 TInt newMode = iScreenSizeMode;
923 TSizeMode* sizeMode = NULL;
926 newMode = (newMode+1)%iModes->Count();
927 sizeMode = (*iModes)[newMode];
929 while (sizeMode==NULL);
930 doSetScreenMode(newMode);
933 void CScreen::doSetScreenMode(TInt aMode,TBool aInsideStartup)
935 WS_ASSERT_DEBUG(IsValidScreenSizeMode(aMode),EWsPanicInvalidScreenSizeMode);
936 TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EScreenSizeModeAboutToChange, aMode));
938 if (iDisplayControl && iDisplayPolicy)
940 TDisplayConfiguration config;
941 TRect sizeModePosition;
943 TSizeMode& mode=*(*iModes)[aMode];
944 config.SetRotation((TDisplayConfiguration::TRotation)mode.iRotation);
945 //This will return suitable config when display is connected
946 //and UI size equal to smallest appmode when disconnected
947 error = iDisplayPolicy->GetSizeModeConfiguration(aMode,config,sizeModePosition);
948 //set appmode in policy
951 iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
953 MWsScene::TSceneRotation oldRotation;
955 config.GetResolution(newUiSize);
956 if (error == KErrNone)
958 oldRotation = iScene->SceneRotation();
959 //The config info will always be set in policy. If display is connected, policy will have
960 //correct composition, Ui and app res. otherwise Ui and appmode will be both set to smallest
961 //app mode, composition will be set to zero
962 if (iFlags&EHasDynamicSizeModes)
964 error = iFallbackMap->Resize(newUiSize);
966 if (error == KErrNone)
968 error = iDisplayControl->SetConfiguration(config);
971 if (error == KErrNone)
973 UpdateDynamicScreenModes();
974 AbortAllDirectDrawing(RDirectScreenAccess::ETerminateScreenMode);
976 if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
978 iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
981 iScreenSizeMode=aMode;
982 //This could fail (to set orientation on the context, or to register the rotated DSA).
983 //Previously, the update rotation was deferred, leaving the size mode out of step with the actual rotation
984 //It also returns false if the orientation is "intentionally" not modified.
985 (void) UpdateOrientation(&oldRotation);
987 //SetDigitiserAreas needs revisiting if/when we support dynamic resolutions
988 //on a screen with touch input.
989 //SetDigitiserAreas(newUiSize);
991 CWsWindowGroup::SetScreenDeviceValidStates(this);
995 iWindowElementSet->ResubmitAllElementExtents();
996 //TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
997 if(BlankScreenOnRotation())
999 iRootWindow->ClearDisplay();
1002 CWsTop::ClearAllRedrawStores();
1003 DiscardAllSchedules();
1004 iRootWindow->InvalidateWholeScreen();
1010 if (iDisplayMapping)
1012 TRect sizeModePosition;
1013 TRAPD(err,sizeModePosition=TRect(OriginL(aMode),ScreenModeSizeInPixelsL(aMode)));
1016 iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
1019 if (!aInsideStartup && (*iModes)[aMode]->iOrigin != (*iModes)[iScreenSizeMode]->iOrigin)
1021 iWindowElementSet->ResubmitAllElementExtents();
1022 if ((*iModes)[aMode]->iRotation == (*iModes)[iScreenSizeMode]->iRotation)
1024 //TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
1025 if(BlankScreenOnRotation())
1027 iRootWindow->ClearDisplay();
1030 CWsTop::ClearAllRedrawStores();
1031 DiscardAllSchedules();
1032 iRootWindow->InvalidateWholeScreen();
1035 iScreenSizeMode=aMode;
1036 //This could fail (to set orientation on the context, or to register the rotated DSA).
1037 //Previously, the update rotation was deferred, leaving the size mode out of step with the actual rotation
1038 //It also returns false if the orientation is not modified.
1039 (void)UpdateOrientation();
1040 CWsWindowGroup::SetScreenDeviceValidStates(this);
1042 TWindowServerEvent::SendScreenDeviceChangedEvents(this);
1046 void CScreen::CycleOrientation()
1048 WS_ASSERT_DEBUG(IsValidScreenSizeMode(iScreenSizeMode),EWsPanicInvalidScreenSizeMode);
1049 TSizeMode& currentSizeMode=*(*iModes)[iScreenSizeMode];
1050 TUint rotations=currentSizeMode.iAlternativeRotations;
1051 TInt currentRotation=currentSizeMode.iRotation;
1052 TInt rotation=currentRotation+1;
1053 while (rotation!=currentRotation)
1055 if (rotation>CFbsBitGc::EGraphicsOrientationRotated270)
1056 rotation=CFbsBitGc::EGraphicsOrientationNormal;
1057 if (ROTATION_TO_FLAG(rotation)&rotations)
1061 if (rotation==currentRotation)
1063 if (rotation>CFbsBitGc::EGraphicsOrientationRotated90)
1068 currentSizeMode.iRotation=REINTERPRET_CAST(CFbsBitGc::TGraphicsOrientation&,rotation);
1069 CWsWindowGroup::NewOrientation(iScreenSizeMode,currentSizeMode.iRotation, iRootWindow);
1071 if (!UpdateOrientation())
1073 // Roll back earlier changes
1074 currentSizeMode.iRotation=REINTERPRET_CAST(CFbsBitGc::TGraphicsOrientation&,currentRotation);
1075 CWsWindowGroup::NewOrientation(iScreenSizeMode, currentSizeMode.iRotation, iRootWindow);
1080 * This method is called either when switching screen size mode, or when the
1081 * orientation of the currently active screen size mode is changed.
1083 TBool CScreen::UpdateOrientation(MWsScene::TSceneRotation* aOldRotation)
1085 CFbsBitGc::TGraphicsOrientation gcOrientation = Orientation();
1087 MWsScene::TSceneRotation oldRotation = aOldRotation? (*aOldRotation):(iScene->SceneRotation());
1088 MWsScene::TSceneRotation newRotation = GcToScreen(gcOrientation);
1089 TDeviceOrientation newDeviceOrientation = GcToDevice(gcOrientation);
1091 // We have to disable the text cursor here while we are still in the
1092 // same orientation or offset as we drew it.
1093 RWsTextCursor* cursor = CWsTop::CurrentTextCursor();
1099 // Some of this method has to be done when changing mode even if not changing rotation
1100 TBool rotating=(oldRotation != newRotation);
1104 // Set the new screen rotation and update the UI element extent
1105 if (iScene->SetSceneRotation(newRotation) != KErrNone)
1107 // Set the new orientation for the DSA device and update the DSA surface
1110 iDsaDevice->SetDeviceOrientation(newDeviceOrientation);
1112 TSurfaceId newSurface;
1113 iDsaDevice->GetSurface(newSurface);
1114 TInt errRegister = iScene->RegisterSurface(newSurface);
1115 WS_ASSERT_DEBUG(KErrNone == errRegister, EWsPanicDirectScreenAccess);
1116 // This will remove all the DSA elements from the scene
1117 AbortAllDirectDrawing(RDirectScreenAccess::ETerminateRotation);
1119 TInt errUnregister = iScene->UnregisterSurface(iDsaSurface);
1120 WS_ASSERT_DEBUG(KErrNone == errUnregister, EWsPanicDirectScreenAccess);
1121 iDsaSurface = newSurface;
1124 //updaterotation should not fail after this point (no cleanup)
1126 //update the last set config with the new rotation change so we don't incorrectly
1127 //change the layer extents
1128 if (iDisplayControl)
1130 TDisplayConfiguration config;
1131 config.SetRotation(static_cast<TDisplayConfiguration::TRotation>(newRotation));
1132 iConfigChangeNotifier->UpdateLastSetConfiguration(config);
1135 TWservCrEvent crEvent(TWservCrEvent::EDeviceOrientationChanged,iScreenNumber,&gcOrientation);
1136 TWindowServerEvent::NotifyDrawer(crEvent);
1138 if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
1140 iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
1145 iRootWindow->AdjustCoordsDueToRotation();
1148 if(BlankScreenOnRotation())
1150 iRootWindow->ClearDisplay();
1153 CWsTop::ClearAllRedrawStores();
1154 DiscardAllSchedules();
1155 iRootWindow->InvalidateWholeScreen();
1160 TPoint CScreen::PhysicalToLogical(TPoint aPhysicalPt)
1162 const TSizeMode& mode=ScreenSizeModeData();
1164 if(!iDisplayMapping)
1167 logicalPt=aPhysicalPt-mode.iOrigin;
1168 if (mode.iScreenScale.iWidth!=1)
1169 logicalPt.iX=(logicalPt.iX>=0 ? logicalPt.iX/mode.iScreenScale.iWidth : (logicalPt.iX-(mode.iScreenScale.iWidth-1))/mode.iScreenScale.iWidth);
1170 if (mode.iScreenScale.iHeight!=1)
1171 logicalPt.iY=(logicalPt.iY>=0 ? logicalPt.iY/mode.iScreenScale.iHeight : (logicalPt.iY-(mode.iScreenScale.iHeight-1))/mode.iScreenScale.iHeight);
1175 //rect with dummy size for coordinates mapping purpose
1176 TRect rectInComp(aPhysicalPt, TSize(1,1));
1178 iDisplayMapping->MapCoordinates(ECompositionSpace, rectInComp, EApplicationSpace,rectInApp);
1179 logicalPt = rectInApp.iTl;
1185 void CScreen::LoadScreenSizesL(TSize aScreenSize)
1187 _LIT(KWSERVNumScrSizeMode, "NUMSCREENMODES");
1188 TBool allowScrGap=WsIniFile->FindVar(iScreenNumber, KWSERVNumScrSizeMode, iNumScreenSizeModes);
1189 iModes=new(ELeave) RPointerArray<TInternalSizeMode>(1);
1190 WS_ASSERT_DEBUG(!allowScrGap || (allowScrGap && iNumScreenSizeModes>0), EWsPanicInvalidScreenSizeMode);
1195 TBuf<32> varNameWidth;
1196 TBuf<32> varNameHeight;
1197 _LIT(KWSERVScreenWidthPattern,"SCR_WIDTH%d");
1198 varNameWidth.Format(KWSERVScreenWidthPattern,screenNum);
1199 _LIT(KWSERVScreenHeightPattern,"SCR_HEIGHT%d");
1200 varNameHeight.Format(KWSERVScreenHeightPattern,screenNum);
1202 if (!WsIniFile->FindVar(iScreenNumber, varNameWidth, screenSize.iWidth) ||
1203 !WsIniFile->FindVar(iScreenNumber, varNameHeight, screenSize.iHeight))
1205 if (allowScrGap && screenNum<=iNumScreenSizeModes)
1207 iModes->AppendL(NULL);
1214 if (screenSize.iWidth==0 && screenSize.iHeight==0)
1216 screenSize=aScreenSize;
1219 if (screenSize.iWidth==-1 && screenSize.iHeight==-1)
1221 screenSize=aScreenSize;
1223 iFlags|=EHasDynamicSizeModes;
1225 TInternalSizeMode* newSizeMode=new(ELeave) TInternalSizeMode(screenSize);
1226 newSizeMode->iFlags|=flags;
1227 CleanupStack::PushL(newSizeMode);
1228 iModes->AppendL(newSizeMode);
1229 CleanupStack::Pop(newSizeMode);
1230 ++iNumSupportedScreenSizeModes;
1232 // If sparse index is enabled and no screen size mode defined, all iModes entries will be NULL
1233 // Otherwise iModes will be empty
1234 if (iModes->Count()==0 || iNumSupportedScreenSizeModes==0)
1236 TInternalSizeMode* defaultSizeMode=new(ELeave) TInternalSizeMode(aScreenSize);
1237 defaultSizeMode->iFlags|=EHalDefault;
1238 if (iModes->Count()>0)
1239 (*iModes)[0]=defaultSizeMode;
1242 CleanupStack::PushL(defaultSizeMode);
1243 iModes->AppendL(defaultSizeMode);
1244 CleanupStack::Pop(defaultSizeMode);
1246 ++iNumSupportedScreenSizeModes;
1249 iNumScreenSizeModes=iNumSupportedScreenSizeModes;
1252 void CScreen::LoadScreenRotationProperties(TInternalSizeMode& aMode, const TInt aModeIndex)
1254 TBuf<32> varRotation;
1255 _LIT(KWSERVScreenRotationPattern,"SCR_ROTATION%d");
1256 varRotation.Format(KWSERVScreenRotationPattern,aModeIndex+1);
1257 TInt rotation=CFbsBitGc::EGraphicsOrientationNormal;
1258 TUint allRotations=0;
1259 TPtrC rotList(NULL,0);
1260 if (WsIniFile->FindVar( iScreenNumber, varRotation,rotList))
1263 TBool foundOne=EFalse;
1268 if (!FindNextValue(lex, rot))
1272 if (rot<0 || rot>360)
1285 if (rot<=CFbsBitGc::EGraphicsOrientationRotated270)
1287 allRotations|=ROTATION_TO_FLAG(rot);
1291 if (allRotations==0)
1292 allRotations=ROTATION_TO_FLAG(rotation);
1293 const TInt KAllRotationsMask = 0xF; //Used to keep the old behaviour
1294 WS_ASSERT_ALWAYS((ROTATION_TO_FLAG(rotation)&KAllRotationsMask)>0, EWsPanicFailedToInitialise);
1295 aMode.iRotation=reinterpret_cast<CFbsBitGc::TGraphicsOrientation&>(rotation);
1296 aMode.iAlternativeRotations=allRotations&KAllRotationsMask;
1299 void CScreen::LoadScreenTwipsProperties(TInternalSizeMode& aMode, const TInt aModeIndex)
1301 TBuf<32> varNameWidth;
1302 _LIT(KWSERVScreenTwipWidthPattern,"SCR_TWIP_WIDTH%d");
1303 varNameWidth.Format(KWSERVScreenTwipWidthPattern,aModeIndex+1);
1304 TBuf<32> varNameHeight;
1305 _LIT(KWSERVScreenTwipHeightPattern,"SCR_TWIP_HEIGHT%d");
1306 varNameHeight.Format(KWSERVScreenTwipHeightPattern,aModeIndex+1);
1309 TBool widthFound = WsIniFile->FindVar(iScreenNumber,varNameWidth,twipsSize.iWidth);
1310 TBool heightFound = WsIniFile->FindVar(iScreenNumber,varNameHeight,twipsSize.iHeight);
1312 // if either of the width or height wsini reads has failed we need to generate default values
1313 switch(aMode.iRotation)
1315 // CFbsBitGc::TGraphicsOrientation
1316 case CFbsBitGc::EGraphicsOrientationRotated90: // deliberate drop-through
1317 case CFbsBitGc::EGraphicsOrientationRotated270:
1319 // CFbsScreenDevice knows nothing about rotation, so we can't use it's PixelsTo Twips methods
1320 // So swap the axis here to use the correct twips per pixel ratio
1322 twipsSize.iWidth = DeviceMap().VerticalPixelsToTwips(aMode.iScreenSize.iWidth);
1324 twipsSize.iHeight = DeviceMap().HorizontalPixelsToTwips(aMode.iScreenSize.iHeight);
1327 case CFbsBitGc::EGraphicsOrientationNormal: // deliberate drop-through
1328 case CFbsBitGc::EGraphicsOrientationRotated180:
1330 twipsSize.iWidth = DeviceMap().HorizontalPixelsToTwips(aMode.iScreenSize.iWidth);
1332 twipsSize.iHeight = DeviceMap().VerticalPixelsToTwips(aMode.iScreenSize.iHeight);
1335 RDebug::Print(_L("** CScreen::LoadScreenTwipsProperties Panic"));
1336 WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
1339 if (widthFound&&heightFound)
1341 aMode.iFlags|=this->ETwipsSpecified;
1343 aMode.iScreenTwipsSize=twipsSize;
1347 void CScreen::LoadScreenSizeProperties(TDisplayMode aDefaultDisplayMode)
1349 for(TInt sizeLoop=0;sizeLoop<iModes->Count();sizeLoop++)
1351 TInternalSizeMode* modePtr=(*iModes)[sizeLoop];
1354 TInternalSizeMode& mode=*modePtr;
1355 TBuf<32> varDisplayMode;
1356 _LIT(KWSERVScreenDisplayModePattern,"SCR_WINDOWMODE%d");
1358 varDisplayMode.Format(KWSERVScreenDisplayModePattern,sizeLoop+1);
1359 mode.iScreenScale.iWidth=1;
1360 mode.iScreenScale.iHeight=1;
1364 _LIT(KWSERVScreenLeftPattern,"SCR_LEFT%d");
1365 _LIT(KWSERVScreenTopPattern,"SCR_TOP%d");
1366 varLeft.Format(KWSERVScreenLeftPattern,sizeLoop+1);
1367 varTop.Format(KWSERVScreenTopPattern,sizeLoop+1);
1368 if (!WsIniFile->FindVar( iScreenNumber, varLeft,mode.iOrigin.iX))
1370 if (!WsIniFile->FindVar( iScreenNumber, varTop,mode.iOrigin.iY))
1373 TPtrC displayModeName(NULL,0);
1374 mode.iDefaultDisplayMode = aDefaultDisplayMode;
1375 // must know rotation before parsing twips
1376 LoadScreenRotationProperties(mode, sizeLoop);
1377 LoadScreenTwipsProperties(mode, sizeLoop);
1380 if(mode.iScreenSize.iWidth == mode.iScreenSize.iHeight && mode.iAlternativeRotations == allRotationsMask)
1382 //square appmode with all four rotations allowed must have square twipsize
1383 if((mode.iFlags&ETwipsSpecified) && mode.iScreenTwipsSize.iWidth != mode.iScreenTwipsSize.iHeight)
1385 RDebug::Print(_L("**Panic: Square appmode with all four rotations must have square twip size"));
1386 WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
1388 //square appmode with all four rotations allowed must have square offset
1389 if(mode.iOrigin.iX != mode.iOrigin.iY)
1391 RDebug::Print(_L("**Panic: Square appmode with all four rotations must have square offset"));
1392 WS_PANIC_ALWAYS(EWsPanicFailedToInitialise);
1397 //Everything else is treated as rectangle appmode. Square appmode not supporting all 4 rotations is considered
1398 //as rectangle appmode as well. Rectangle appmode suports 2 rotations at most (0 and 180, or 90 and 270)
1399 //first rotation of the appmode is taken to apply the corresponding rotation mask
1400 if(!((mode.iAlternativeRotations&KRotation0_180Mask) == mode.iAlternativeRotations
1401 || (mode.iAlternativeRotations&KRotation90_270Mask) == mode.iAlternativeRotations))
1403 RDebug::Print(_L("**Panic_DEBUG: non square appmode can only define (0,180) or (90,270) rotations"));
1404 WS_PANIC_DEBUG(EWsPanicFailedToInitialise);
1405 //in relase build, no panic, just correct the rotations set
1408 //correct the rotations set
1409 mode.iAlternativeRotations &= ((ROTATION_TO_FLAG(mode.iRotation) & KRotation0_180Mask)? KRotation0_180Mask:
1410 KRotation90_270Mask);
1415 TInt intForFindVar=0;
1416 _LIT(KWSERVIniFileVarSizeMode,"SIZE_MODE");
1417 WsIniFile->FindVar( iScreenNumber, KWSERVIniFileVarSizeMode,intForFindVar);
1418 iSizeEnforcementMode=(TScreenModeEnforcement)intForFindVar;
1421 void CScreen::SetDigitiserAreas(const TSize& aUiSize)
1422 { //aUiSize should be the unrotated current ui size
1423 //SetDigitiserAreas needs revisiting if/when we support dynamic resolutions on a screen
1424 //with touch input. It is not known how digitiser coordinates will be represented if the
1425 //physical display resolution changes. Currently digitisers are only supported on screen 0,
1426 //and dynamic resolution only applies to higher screen numbers on real hardware.
1427 for(TInt sizeLoop=0;sizeLoop<iModes->Count();sizeLoop++)
1429 TInternalSizeMode* modePtr=(*iModes)[sizeLoop];
1432 TInternalSizeMode& mode=*modePtr;
1433 if(mode.iFlags & EClientDefinedDigitiserArea)
1435 //if it's client set, keep it unchanged.
1438 switch (mode.iRotation)
1440 case CFbsBitGc::EGraphicsOrientationNormal:
1441 mode.iPointerCursorArea=iDigitiserArea;
1443 case CFbsBitGc::EGraphicsOrientationRotated90:
1444 mode.iPointerCursorArea.SetRect(iDigitiserArea.iTl.iY,aUiSize.iWidth-iDigitiserArea.iBr.iX,
1445 iDigitiserArea.iBr.iY,aUiSize.iWidth-iDigitiserArea.iTl.iX);
1447 case CFbsBitGc::EGraphicsOrientationRotated180:
1448 mode.iPointerCursorArea.SetRect(-(iDigitiserArea.iBr-aUiSize),-(iDigitiserArea.iTl-aUiSize));
1450 case CFbsBitGc::EGraphicsOrientationRotated270:
1451 mode.iPointerCursorArea.SetRect(aUiSize.iHeight-iDigitiserArea.iBr.iY,iDigitiserArea.iTl.iX,
1452 aUiSize.iHeight-iDigitiserArea.iTl.iY,iDigitiserArea.iBr.iX);
1455 WS_PANIC_ALWAYS(EWsPanicInvalidRotation);
1460 void CScreen::GetScreenSizeAndRotation(TPixelsTwipsAndRotation &aSar, TInt aScreenMode)
1462 TSizeMode& mode=*(*iModes)[aScreenMode];
1463 aSar.iRotation=mode.iRotation;
1464 aSar.iPixelSize=mode.iScreenSize;
1465 aSar.iTwipsSize=mode.iScreenTwipsSize;
1466 if (aSar.iTwipsSize.iWidth==0)
1468 aSar.iTwipsSize.iWidth = iDeviceMap->HorizontalPixelsToTwips(aSar.iPixelSize.iWidth);
1469 aSar.iTwipsSize.iHeight = iDeviceMap->VerticalPixelsToTwips(aSar.iPixelSize.iHeight);
1473 void CScreen::GetScreenSizeAndRotation(TPixelsAndRotation &aSar, TInt aScreenMode)
1475 TSizeMode& mode=*(*iModes)[aScreenMode];
1476 aSar.iRotation=mode.iRotation;
1477 aSar.iPixelSize=mode.iScreenSize;
1480 TBool CScreen::SetScreenModeEnforcement(TInt aMode)
1482 if (aMode<0 || aMode>ESizeEnforcementPixelsTwipsAndRotation)
1484 TScreenModeEnforcement newMode=(TScreenModeEnforcement)aMode;
1485 if (newMode!=iSizeEnforcementMode)
1487 iSizeEnforcementMode=newMode;
1488 CWsWindowGroup::SetScreenDeviceValidStates(this);
1494 void CScreen::IncContrast()
1497 if (iMaxContrast<0) //If failed to get it sofar get it again
1498 TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Get(iScreenNumber,HALData::EDisplayContrastMax,iMaxContrast));
1499 if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Get(iScreenNumber,HALData::EDisplayContrast,contrast)))
1501 if (contrast==iMaxContrast)
1503 TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Set(iScreenNumber,HALData::EDisplayContrast,++contrast));
1506 void CScreen::DecContrast()
1509 if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Get(iScreenNumber,HALData::EDisplayContrast,contrast)))
1513 if (iMaxContrast<0 && TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast,
1514 HAL::Get(iScreenNumber,HALData::EDisplayContrastMax,iMaxContrast)))
1516 contrast=iMaxContrast+1;
1518 TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EContrast, HAL::Set(iScreenNumber,HALData::EDisplayContrast,--contrast));
1521 void CScreen::IncBrightness()
1524 if (iMaxBrightness<0) //If failed to get it sofar get it again
1525 TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(iScreenNumber,HALData::EDisplayBrightnessMax,iMaxBrightness));
1526 if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(iScreenNumber,HALData::EDisplayBrightness,brightness)))
1528 if (brightness==iMaxBrightness)
1530 TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(iScreenNumber,HALData::EDisplayBrightness,++brightness));
1533 void CScreen::DecBrightness()
1536 if (TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Get(iScreenNumber,HALData::EDisplayBrightness,brightness)))
1540 if (iMaxBrightness<0 && TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight,
1541 HAL::Get(iScreenNumber,HALData::EDisplayBrightnessMax,iMaxBrightness)))
1543 brightness=iMaxBrightness+1;
1545 TWindowServerEvent::ProcessErrorMessages(TWsErrorMessage::EBackLight, HAL::Set(iScreenNumber,HALData::EDisplayBrightness,--brightness));
1547 TInt CScreen::GetScreenSizeModeList(RArray<TInt>& aList) const
1550 TInt numModes=iNumScreenSizeModes;
1551 TInt err=aList.Reserve(numModes);
1555 for (index=0; index<numModes; ++index)
1557 TSizeMode* modePtr=(*iModes)[index];
1559 aList.Append(index); //Can't fail due to reserve
1561 TInt count=aList.Count();
1565 TInt CScreen::GetScreenSizeModeListL()
1568 CleanupClosePushL(list);
1569 TInt count=GetScreenSizeModeList(list);
1570 User::LeaveIfError(count);
1571 CWsClient::ReplyBuf(&list[0], count*sizeof(TInt));
1572 CleanupStack::PopAndDestroy(&list);
1576 void CScreen::SetInitialScreenSizeModeAndRotation()
1577 { //Set first app mode that supports current supported rotations if available
1579 TInt firstMode = -1;
1581 // Since all screen rotation modes are supported.
1582 TBitFlags32 rotationFlags = CFbsBitGc::EGraphicsOrientationNormal|CFbsBitGc::EGraphicsOrientationRotated90|CFbsBitGc::EGraphicsOrientationRotated180|CFbsBitGc::EGraphicsOrientationRotated270;
1583 for (index=0; index<iModes->Count(); ++index)
1585 TSizeMode* modePtr=(*iModes)[index];
1588 if (firstMode == -1)
1592 if (rotationFlags.IsSet((TInt)modePtr->iRotation))
1600 if (bestMode != -1) //found a mode that supports current supported rotations
1602 iScreenSizeMode = bestMode;
1606 if (firstMode != -1) //could only find a mode that doesnt support current supported rotations
1608 iScreenSizeMode = firstMode;
1612 return; //couldn't find a mode at all
1617 iDisplayPolicy->SetLastAppMode(iScreenSizeMode);
1620 SetDigitiserAreas(iScreenDevice->SizeInPixels()); //unrotated size in pixels
1622 // Here we are mixing CFbsBitGc::TGraphicsOrientation with MWsScene::TSceneRotation
1623 // As they both have same values it is fine for now
1624 iScene->SetSceneRotation(GcToScreen(ScreenSizeModeData().iRotation)); //set rotation
1628 TDisplayMode CScreen::FirstDefaultDisplayMode() const
1631 while ((*iModes)[++mode]==NULL)
1633 WS_ASSERT_DEBUG(mode<iModes->Count()-1,EWsPanicInvalidScreenSizeMode);
1635 return((*iModes)[mode]->iDefaultDisplayMode);
1638 void CScreen::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
1640 iRedraw->AddRedrawRegion(aRegion, aSchedule, aDepth);
1643 void CScreen::ScheduleRender(const TTimeIntervalMicroSeconds& aFromNow)
1645 iRedraw->ScheduleRender(aFromNow);
1648 void CScreen::DoRedrawNow()
1650 iRedraw->DoRedrawNow();
1653 // See CScreenRedraw::IsUpdatePending() for important notes on usage.
1654 void CScreen::RedrawNowIfPending()
1656 if(iRedraw->IsUpdatePending())
1660 TBool CScreen::IsQuickFadeScheduled( CWsWindow* aWin ) const
1662 return iRedraw->IsQuickFadeScheduled( aWin );
1665 void CScreen::RemoveFromQuickFadeList( CWsWindow* aWin )
1667 iRedraw->RemoveFromQuickFadeList( aWin );
1670 void CScreen::AcceptFadeRequest( CWsWindow* aWin, TBool aIsFaded )
1672 iRedraw->AcceptFadeRequest( aWin, aIsFaded );
1675 // implementing MWsScreen
1677 const TTime& CScreen::Now() const
1679 return iRedraw->Now();
1682 void CScreen::ScheduleAnimation(TAnimType aType, const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow,const TTimeIntervalMicroSeconds& aFreq,const TTimeIntervalMicroSeconds& aStop, CWsWindow* aWindow)
1684 iRedraw->ScheduleAnimation(aType, aRect,aFromNow,aFreq,aStop, aWindow);
1687 TBool CScreen::IsScheduled(TAnimType aType, const TRect& aRect, CWsWindow* aWindow) const
1689 return iRedraw->IsScheduled(aType, aRect, aWindow);
1692 void CScreen::OnAnimation(TRequestStatus* aFinished)
1694 iRedraw->OnAnimation(aFinished);
1697 void CScreen::Redraw()
1699 STACK_REGION bounds;
1700 bounds.AddRect(DrawableArea());
1701 AddRedrawRegion(bounds);
1705 TBool CScreen::RedrawInvalid(const TArray<TGraphicDrawerId>& aInvalid)
1707 TBool wasDirty = EFalse;
1708 STACK_REGION bounds;
1709 bounds.AddRect(DrawableArea());
1711 TWalkWindowTreeCalcInvalidGraphics calc(&bounds,dirty,aInvalid);
1712 if(calc.CreateSubRegion())
1714 calc.CalcInvalid(*this);
1715 if(dirty.CheckError() || dirty.Count())
1720 calc.DestroyRegions();
1728 Overidding MWsObjectProvider
1730 TAny* CScreen::ResolveObjectInterface(TUint aTypeId)
1732 TAny* interface = NULL;
1736 case MWsWindow::EWsObjectInterfaceId:
1737 interface = static_cast<MWsWindow*>(RootWindow());
1739 case MWsScreenConfigList::EWsObjectInterfaceId:
1740 interface = static_cast<MWsScreenConfigList*>(this);
1742 case MWsScreenConfig::EWsObjectInterfaceId:
1743 interface = static_cast<MWsScreenConfig*>(this);
1745 case MWsWindowTree::EWsObjectInterfaceId:
1746 interface = static_cast<MWsWindowTree*>(this);
1750 interface = iRedraw->ResolveObjectInterface(aTypeId);
1755 void CScreen::SendTree() const
1757 if(!iWindowTreeObserver)
1760 TWalkWindowTreeSendState wtw(*iWindowTreeObserver);
1761 RootWindow()->WalkWindowTreeBackToFront(wtw, EVisitParentNodesFirst);
1763 //Standard text cursors
1764 RWsTextCursor* cursor = CWsTop::CurrentTextCursor();
1766 cursor->SendState(*iWindowTreeObserver);
1769 SpriteManager()->SendState(*iWindowTreeObserver);
1771 //Window Group Chains
1772 for(CWsWindowGroup *group=RootWindow()->Child(); group!=NULL; group=group->NextSibling())
1774 group->SendStateWindowGroupChain(*iWindowTreeObserver);
1778 TDisplayMode CScreen::DisplayMode() const
1780 return iScreenDevice->DisplayMode();
1783 TSize CScreen::SizeInPixels() const
1785 return iScreenDevice->SizeInPixels();
1788 TSize CScreen::SizeInTwips() const
1790 return iScreenDevice->SizeInTwips();
1793 void CScreen::DiscardAllSchedules()
1795 iRedraw->DiscardAllSchedules();
1798 void CScreen::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
1800 iRedraw->ScheduleRegionUpdate(aDefinitelyDirty);
1803 TBool CScreen::IsDSAClientWindow( const CWsClientWindow* aWin ) const
1806 if ( ! iDirects.IsEmpty() )
1808 TSglQueIter<CWsDirectScreenAccess> iter( (TSglQueBase&)iDirects );
1810 CWsDirectScreenAccess* dsa;
1811 while ( (dsa = iter++) != NULL && !res )
1813 res = (dsa->ClientWindow() == aWin) && (dsa->IsVisible());
1820 Update the UI element composition method based on whether
1821 there are any externals surfaces present and the current display mode.
1822 If the method changes, recomposition is triggered.
1824 void CScreen::UpdateCompositionMode()
1829 void CScreen::ElementAdded()
1831 UpdateCompositionMode();
1834 void CScreen::ElementRemoved()
1836 UpdateCompositionMode();
1839 CRegisteredSurfaceMap* CScreen::SurfaceMap()
1844 void CScreen::InitializeSceneL()
1846 // Ensure the surface is not valid to start with.
1847 iDsaSurface = TSurfaceId::CreateNullId();
1848 iWindowElementSet = CWindowElementSet::NewL(*iScene);
1851 MWsElement* CScreen::CreateUiElementL(const TRect& aExtent)
1853 MWsElement* pElement = iScene->CreateSceneElementL();
1856 pElement->GetRenderStageFlags(flags);
1857 flags |= MWsElement::EElementIsIndirectlyRenderedUserInterface;
1858 pElement->SetRenderStageFlags(flags);
1860 iScene->InsertSceneElement(pElement, NULL);
1862 pElement->SetDestinationRectangle(aExtent);
1863 pElement->SetSourceRectangle(aExtent); //initial guess... updated by PositionUiElements
1868 void CScreen::InitializeUiElementsL()
1870 const TRect screenRect(iScreenDevice->SizeInPixels());
1871 MWsElement* pElement = CreateUiElementL(screenRect);
1876 pElement->GetTargetRendererFlags(flags);
1877 flags |= MWsElement::EElementTransparencySource;
1878 pElement->SetTargetRendererFlags(flags);
1882 TInt CScreen::InitializeDsaSurface()
1884 WS_ASSERT_DEBUG(iDsaSurface.IsNull(),EWsPanicInvalidOperation);
1885 iDsaDevice->GetSurface(iDsaSurface);
1886 // Currently Surface Manager does not recognise DSA surface IDs originating
1887 // from the Screen Driver. This causes it to fail to register such
1888 // surfaces. OpenWF should be amended to properly register DSA surfaces.
1889 iScene->RegisterSurface(iDsaSurface);
1894 TSize CScreen::DSASizeInPixels() const
1898 return iDsaDevice->SizeInPixels();
1906 MWsTextCursor* CScreen::RenderStageTextCursor() const
1908 return iRedraw->RenderStageTextCursor();
1911 void CScreen::ClearDsaSurface(const TRect& area, const TRgb& color)
1913 WS_ASSERT_DEBUG(iDsaGc, EWsPanicInvalidOperation);
1914 iDsaGc->SetBrushStyle(CFbsBitGc::ESolidBrush);
1915 iDsaGc->SetPenStyle(CFbsBitGc::ENullPen);
1916 iDsaGc->SetBrushColor(color);
1917 iDsaGc->DrawRect(area);
1918 iDsaDevice->Update();
1921 void CScreen::ReleaseDsaScreenDevice()
1923 //This function checks if any of the DSA currently active on the screen is actually used to draw
1924 //If not it unregister the DSA surface and destroys the iDsaDevice.
1925 //This function should be called only by a drawing DSA so a surface should be in place.
1926 iNumberDrawingDsa--;
1927 if(iNumberDrawingDsa == 0)
1929 WS_ASSERT_DEBUG(!iDsaSurface.IsNull(),EWsPanicInvalidOperation);
1932 // Currently Surface Manager does not recognise DSA surface IDs originating
1933 // from the Screen Driver. This causes it to fail to unregister such
1934 // surfaces. OpenWF should be amended to properly register DSA surfaces.
1935 iScene->UnregisterSurface(iDsaSurface);
1939 //the old surface Id is now meaningless
1940 iDsaSurface = TSurfaceId::CreateNullId();
1944 TInt CScreen::SetConfiguration(const TDisplayConfiguration& aConfigInput)
1946 TInt reply = KErrNone;
1949 TDisplayConfiguration config(aConfigInput);
1950 TRect sizeModePosition;
1952 { //validate config and update to a valid hardware config
1953 reply = iDisplayPolicy->GetSizeModeConfiguration(iScreenSizeMode,config,sizeModePosition);
1954 if (reply >= KErrNone)
1955 {//set appmode in policy
1956 if (iDisplayMapping)
1958 iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
1963 { //exessive strategy: limit rotation agains curr app mode.
1964 //really we want the system to accept the rotation change regardless of the app mode.
1965 TDisplayConfiguration::TRotation newRot;
1966 if (aConfigInput.GetRotation(newRot))
1967 { //This should cast between rotation enumertaions "properly"
1968 if (!(iModes[0][iScreenSizeMode]->iAlternativeRotations&(1<<newRot)))
1974 if (reply < KErrNone)
1978 MWsScene::TSceneRotation oldRotation;
1979 oldRotation = iScene->SceneRotation();
1981 config.GetResolution(newUiSize);
1982 TDisplayConfiguration oldConfig;
1983 iDisplayControl->GetConfiguration(oldConfig);
1984 if(iFlags&EHasDynamicSizeModes)
1986 reply = iFallbackMap->Resize(newUiSize);
1988 if (reply >= KErrNone)
1990 reply=iDisplayControl->SetConfiguration(config);
1992 if (reply==KErrNone)
1995 oldConfig.GetResolution(oldConfigRes);
1996 TDisplayConfiguration newConfig;
1997 if (oldConfigRes.iWidth == 0 || oldConfigRes.iHeight == 0)
1999 iDisplayControl->GetConfiguration(newConfig);
2000 RecalculateModeTwips(&newConfig); //needs res and twips information
2002 UpdateDynamicScreenModes();
2004 //update the last set config in the config change notifier to
2005 //prevent SetConfiguration() from being called again!
2006 newConfig.ClearAll();
2007 iDisplayControl->GetConfiguration(newConfig);
2008 iConfigChangeNotifier->UpdateLastSetConfiguration(newConfig);
2010 TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EScreenSizeModeAboutToChange, iScreenSizeMode));
2011 // This will remove all the DSA elements from the scene
2012 AbortAllDirectDrawing(RDirectScreenAccess::ETerminateRotation);
2014 //SetDigitiserAreas needs revisiting if/when we support dynamic resolutions
2015 //on a screen with touch input.
2016 //SetDigitiserAreas(newUiSize);
2018 //failure here should only be because of DSA orientation change failure, which shouldn't happen, either.
2019 //Or there may be no change to do.
2020 (void)UpdateOrientation(&oldRotation);
2022 iWindowElementSet->ResubmitAllElementExtents();
2023 if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
2025 iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
2028 iRootWindow->AdjustCoordsDueToRotation();
2030 //TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
2031 if(BlankScreenOnRotation())
2033 iRootWindow->ClearDisplay();
2036 CWsTop::ClearAllRedrawStores();
2037 DiscardAllSchedules();
2038 iRootWindow->InvalidateWholeScreen();
2039 CWsWindowGroup::SetScreenDeviceValidStates(this);
2040 TWindowServerEvent::SendScreenDeviceChangedEvents(this);
2049 reply = KErrNotSupported;
2055 * Updates the screen device display properties. This is to ensure the screen device is
2056 * consistent with any configuration changes not made using CScreen::SetConfiguration.
2058 * @param aConfigInput a fully populated display configuration
2060 TInt CScreen::UpdateConfiguration(const TDisplayConfiguration& aConfigInput)
2062 TInt reply = KErrNone;
2065 TDisplayConfiguration config(aConfigInput);
2066 TRect sizeModePosition;
2068 { //validate config and update to a valid hardware config
2069 reply = iDisplayPolicy->GetSizeModeConfiguration(iScreenSizeMode,config,sizeModePosition);
2070 if (reply >= KErrNone)
2071 {//set appmode in policy
2072 if (iDisplayMapping)
2074 iDisplayMapping->SetSizeModeExtent(sizeModePosition,MWsDisplayMapping::KOffsetAll);
2079 { //exessive strategy: limit rotation agains curr app mode.
2080 //really we want the system to accept the rotation change regardless of the app mode.
2081 TDisplayConfiguration::TRotation newRot;
2082 if (aConfigInput.GetRotation(newRot))
2083 { //This should cast between rotation enumertaions "properly"
2084 if (!(iModes[0][iScreenSizeMode]->iAlternativeRotations&(1<<newRot)))
2091 MWsScene::TSceneRotation oldRotation;
2092 oldRotation = iScene->SceneRotation();
2094 config.GetResolution(newUiSize);
2095 if(iFlags&EHasDynamicSizeModes)
2097 reply = iFallbackMap->Resize(newUiSize);
2100 RecalculateModeTwips(&config); //needs res and twips information
2101 UpdateDynamicScreenModes();
2103 TWindowServerEvent::NotifyDrawer(TWservCrEvent(TWservCrEvent::EScreenSizeModeAboutToChange, iScreenSizeMode));
2104 // This will remove all the DSA elements from the scene
2105 AbortAllDirectDrawing(RDirectScreenAccess::ETerminateRotation);
2107 //SetDigitiserAreas needs revisiting if/when we support dynamic resolutions
2108 //on a screen with touch input.
2109 //SetDigitiserAreas(newUiSize);
2111 //failure here should only be because of DSA orientation change failure, which shouldn't happen, either.
2112 //Or there may be no change to do.
2113 (void)UpdateOrientation(&oldRotation);
2115 iWindowElementSet->ResubmitAllElementExtents();
2116 if(iDsaDevice && iDsaDevice->GraphicsAccelerator())
2118 iDsaDevice->ChangeScreenDevice(iDsaDevice); // orientation has changed, therefore we need to re-initialise the screen device's graphics accelerator
2121 iRootWindow->AdjustCoordsDueToRotation();
2123 //TODO jonas: we'd like to not have to clear at all... make the actual change to compositor etc lazily!
2124 if(BlankScreenOnRotation())
2126 iRootWindow->ClearDisplay();
2129 CWsTop::ClearAllRedrawStores();
2130 DiscardAllSchedules();
2131 iRootWindow->InvalidateWholeScreen();
2132 CWsWindowGroup::SetScreenDeviceValidStates(this);
2133 TWindowServerEvent::SendScreenDeviceChangedEvents(this);
2137 reply = KErrNotSupported;
2142 void CScreen::UpdateDynamicScreenModes()
2144 WS_ASSERT_DEBUG(iDisplayControl,EWsPanicNoDisplayControl);
2145 TDisplayConfiguration newConfig;
2146 iDisplayControl->GetConfiguration(newConfig);
2149 newConfig.GetResolution(res);
2150 newConfig.GetResolutionTwips(twips);
2151 for (TInt i=0; i<iModes->Count(); i++)
2153 if ((*iModes)[i] && ((*iModes)[i]->iFlags & EDynamic))
2155 (*iModes)[i]->iScreenSize = res;
2156 (*iModes)[i]->iScreenTwipsSize = twips;
2161 void CScreen::RecalculateModeTwips(const TDisplayConfiguration* aConfig)
2163 TDisplayConfiguration config;
2164 iDisplayControl->GetConfiguration(config);
2167 if (aConfig) //called from SetConfiguration
2169 aConfig->GetResolution(res);
2170 if (res.iWidth == 0 || res.iHeight == 0)
2174 aConfig->GetResolutionTwips(twips);
2176 else //called from DisplayChangeNotifier during attach
2178 config.GetResolution(res);
2179 if ((res.iWidth == 0 || res.iHeight == 0) && !iDisplayPolicy)
2183 config.GetResolutionTwips(twips);
2187 for (TInt ii=0; ii<iModes->Count(); ii++)
2189 TRAP(err, flags = ModePtrL(ii)->iFlags);
2190 if (err != KErrNone || flags&(ETwipsSpecified|EDynamic))
2191 { //continue if mode doesnt exist,twips specified or dynamic mode specified
2196 { //get ideal config for app mode from policy
2199 TInt err = iDisplayPolicy->GetSizeModeConfiguration(ii,config,modePosition);
2200 if (err != KErrNone)
2201 { //nothing we can do, the twips will not be calculated correctly
2204 config.GetResolution(res);
2205 config.GetResolutionTwips(twips);
2207 TSizeMode* modePtr=(*iModes)[ii];
2208 modePtr->iScreenTwipsSize.iWidth = (twips.iWidth * modePtr->iScreenSize.iWidth)/
2210 modePtr->iScreenTwipsSize.iHeight = (twips.iHeight * modePtr->iScreenSize.iHeight)/
2216 TInt CScreen::AddNotificationClient(CWsClient *aClient)
2218 TInt err = iWsClientList.InsertInAddressOrder(aClient);
2219 if(!(err == KErrNone || err == KErrAlreadyExists))
2225 void CScreen::RemoveNotificationClient(CWsClient *aClient)
2227 TInt index = iWsClientList.FindInAddressOrder(aClient);
2228 if(index != KErrNotFound)
2230 iWsClientList.Remove(index);
2233 TInt CScreen::GetNotificationClients(RPointerArray<CWsClient>& aClientsArray)
2235 TInt err = aClientsArray.Reserve(iWsClientList.Count());
2239 for(TInt i = 0; i < iWsClientList.Count(); i++)
2241 aClientsArray.Append(iWsClientList[i]);
2246 TInt CScreen::FindNotificationClient (CWsClient *aClient)
2248 return iWsClientList.FindInAddressOrder(aClient);
2251 // implementing MWsScreenConfig... this might be better as RS interface
2252 TSize CScreen::ScreenModeSizeInPixels() const
2254 return (*iModes)[iScreenSizeMode]->iScreenSize;
2256 TInt CScreen::Stride() const
2260 TInt CScreen::SizeMode() const
2262 return iScreenSizeMode;
2265 TSize CScreen::ScalingFactor() const
2267 return (*iModes)[iScreenSizeMode]->iScreenScale;
2269 TPoint CScreen::Origin() const
2271 return (*iModes)[iScreenSizeMode]->iOrigin;
2273 TPoint CScreen::ScaledOrigin() const
2275 return (*iModes)[iScreenSizeMode]->ScaledOrigin();
2278 const CScreen::TInternalSizeMode* CScreen::ModePtrL(TInt aIndex) const
2280 if (aIndex>=iModes->Count() || aIndex<0)
2282 User::Leave(KErrArgument);
2286 User::Leave(KErrNotReady);
2288 TInternalSizeMode* modePtr=(*iModes)[aIndex];
2291 User::Leave(KErrArgument);
2296 TDisplayMode CScreen::DisplayModeL(TInt aIndex) const
2298 return ModePtrL(aIndex)->iDefaultDisplayMode;
2300 TSize CScreen::ScreenModeSizeInPixelsL(TInt aIndex) const
2302 return ModePtrL(aIndex)->iScreenSize;
2304 TSize CScreen::ScreenModeSizeInTwipsL(TInt aIndex) const
2306 return ModePtrL(aIndex)->iScreenTwipsSize;
2309 CFbsBitGc::TGraphicsOrientation CScreen::OrientationL(TInt aIndex) const
2311 return ModePtrL(aIndex)->iRotation;
2313 TInt CScreen::AvailableOrientationsL(TInt aIndex) const
2315 return ModePtrL(aIndex)->iAlternativeRotations;
2317 TSize CScreen::ScalingFactorL(TInt aIndex) const
2319 return ModePtrL(aIndex)->iScreenScale;
2321 TPoint CScreen::OriginL(TInt aIndex) const
2323 return ModePtrL(aIndex)->iOrigin;
2325 TPoint CScreen::ScaledOriginL(TInt aIndex) const
2327 return ModePtrL(aIndex)->ScaledOrigin();
2329 TInt CScreen::ModeFlagsL(TInt aIndex) const
2331 return ModePtrL(aIndex)->iFlags;
2333 void CScreen::SetCurrentScreenModeAttributes(const TSizeMode &aModeData)
2335 TSizeMode* modeToOverwrite=(*iModes)[iScreenSizeMode];
2336 *modeToOverwrite=aModeData;
2339 void CScreen::ScheduleWindow(CWsWindow* aWindow)
2341 iRedraw->ScheduleWindow(aWindow);
2344 void CScreen::RemoveFromScheduledList(CWsWindow* aWindow)
2346 iRedraw->RemoveFromScheduledList(aWindow);
2349 void CScreen::RemoveFromTimedDrawList(CWsWindow* aWindow)
2351 iRedraw->RemoveFromTimedDrawList(aWindow);
2354 void CScreen::SetupVisibleRegionTracking(CWsWindow& aWindow, TBool aRegister) const
2356 if(ChangeTracking() && iWindowVisibilityNotifier)
2360 iWindowVisibilityNotifier->RegisterWindow(aWindow);
2364 iWindowVisibilityNotifier->UnregisterWindow(aWindow);
2369 TBool CScreen::IsAnimating() const
2371 return iRedraw->IsAnimating();