Update contrib.
1 // Copyright (c) 2008-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.
20 #include "wsdisplaychangeao.h"
24 CWsDisplayChangeNotifier::CWsDisplayChangeNotifier(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
25 :CActive(EEventPriority), iNextLevelInterface(aNextLevelInterface), iOwner(aOwner)
27 CActiveScheduler::Add(this);
30 CWsDisplayChangeNotifier* CWsDisplayChangeNotifier::NewL(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
32 CWsDisplayChangeNotifier* self = new(ELeave) CWsDisplayChangeNotifier(aNextLevelInterface, aOwner);
33 CleanupStack::PushL(self);
39 void CWsDisplayChangeNotifier::ConstructL()
41 iRetry = CEventQueueRetry::NewL();
44 void CWsDisplayChangeNotifier::IssueNotificationRequest()
46 //before submitting request, save the last resolution list
47 iLastErr = iNextLevelInterface->GetResolutions(iLastResList);
48 if(iLastErr != KErrNone)
50 //probabally KErrDisconnect
53 iNextLevelInterface->NotifyOnDisplayChange(iStatus);
57 void CWsDisplayChangeNotifier::CancelNotificationRequest()
62 void CWsDisplayChangeNotifier::RunL()
64 if(iStatus == KErrNone)
66 //IssueNotificationRequest() will overwrite iLastResList and iLastErr, save a copy first
67 TInt lastErr = iLastErr;
68 RArray<MWsDisplayControl::TResolution> lastResList;
69 CleanupClosePushL(lastResList);
70 lastResList.ReserveL(iLastResList.Count());
71 for(TInt i = 0; i < iLastResList.Count(); i++)
73 lastResList.Append(iLastResList[i]);
75 //submit request again, this should be called as early as possible in RunL(). Otherwise display change occurs
76 //before submitting the request will be missed.
77 IssueNotificationRequest();
79 RArray<MWsDisplayControl::TResolution> currentResList;
80 CleanupClosePushL(currentResList);
81 TInt err = iNextLevelInterface->GetResolutions(currentResList);
84 //probabally KErrDisconnect
85 currentResList.Reset();
87 //return code not equal also counts as an event
88 if(!IsResListEqual(currentResList, lastResList) || lastErr != err)
90 //first thing is to cancel the retry AO if it's running for the last event
91 iRetry->CancelRetry();
93 iOwner->IncreaseDisplaySpinner();
94 //put display change event on queue
95 RPointerArray<CWsClient> clientArray;
96 CleanupClosePushL(clientArray);
97 User::LeaveIfError(iOwner->GetNotificationClients(clientArray));
98 TBool eventOnAllQueues = ETrue;
99 for(TInt i = 0; i < clientArray.Count(); i++)
101 if(!TWindowServerEvent::SendDisplayChangedEvents(clientArray[i], iOwner->ScreenNumber(),
102 iOwner->ConfigSpinner(), iOwner->DisplaySpinner()))
104 eventOnAllQueues = EFalse;
105 clientArray[i]->SetRetryFlag(EEventDisplayChanged);
108 CleanupStack::PopAndDestroy(&clientArray);
110 //some event queues are full, kick off retry AO
111 if(!eventOnAllQueues)
113 iRetry->Init(iOwner);
114 iRetry->Retry(KRetryInitialDelay);
117 //codes below are dealing with detach/attach
118 MWsDisplayPolicy* policy = iOwner->DisplayPolicy();
119 if((err == KErrDisconnected && lastErr != KErrDisconnected)
120 ||(currentResList.Count() == 0 && lastResList.Count() != 0))
122 //The display is disconnected
125 TInt appMode = policy->SuitableAppMode(MWsDisplayPolicy::EDetach);
126 //Last app mode can be resumed when display is connected again from disconnection
127 iOwner->DisplayPolicy()->SetLastAppMode(iOwner->ScreenSizeMode());
128 //This gonna set the screen mode to smallest app mode
131 iOwner->doSetScreenMode(appMode);
136 //if policy is not available, do a SetConfiguration set to the same config before disconnect
137 //this will update parameters for MDisplayMapping stored in CDisplayPolicy
138 TDisplayConfiguration config;
139 iNextLevelInterface->GetConfiguration(config);
140 iNextLevelInterface->SetConfiguration(config);
142 //stop DSA drawing as the above set screen mode is under disconnection and it won't
143 //go the normal routine to stop DSA drawing
144 iOwner->AbortAllDirectDrawing(RDirectScreenAccess::ETerminateScreenMode);
147 if(currentResList.Count() > 0 && lastResList.Count() == 0)
149 //The display is connected
152 TInt appMode = policy->SuitableAppMode(MWsDisplayPolicy::EAttach);
153 //This gonna resume the screen mode to the one before disconnection
156 iOwner->doSetScreenMode(appMode);
161 //if policy is not available, force a SetConfiguration to trigger a config change notification
162 //as the twips size in config is changed on attaching display though resolution may remain 0x0
163 TDisplayConfiguration config;
164 config.SetResolution(currentResList[0].iPixelSize);
165 iOwner->SetConfiguration(config);
167 iOwner->RecalculateModeTwips();
170 CleanupStack::PopAndDestroy(2, &lastResList);
172 else if(iStatus != KErrCancel && iStatus != KErrNotSupported)
174 IssueNotificationRequest(); //This should be ok, not deadlock
179 TBool CWsDisplayChangeNotifier::IsResListEqual(RArray<MWsDisplayControl::TResolution>& aResListA, RArray<MWsDisplayControl::TResolution>& aResListB)
181 if (aResListA.Count() != aResListB.Count())
184 for(TInt i = 0; i < aResListA.Count(); i++)
186 if(aResListA[i].iPixelSize != aResListB[i].iPixelSize)
190 if(aResListA[i].iTwipsSize != aResListB[i].iTwipsSize)
194 if(!(aResListA[i].iFlags == aResListB[i].iFlags))
203 void CWsDisplayChangeNotifier::DoCancel()
206 CWsDisplayChangeNotifier::~CWsDisplayChangeNotifier()
209 iLastResList.Close();
213 CWsConfigChangeNotifier::CWsConfigChangeNotifier(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
214 :CActive(EEventPriority), iNextLevelInterface(aNextLevelInterface), iOwner(aOwner)
216 CActiveScheduler::Add(this);
219 CWsConfigChangeNotifier* CWsConfigChangeNotifier::NewL(MWsDisplayControl* aNextLevelInterface, CScreen *aOwner)
221 CWsConfigChangeNotifier* self = new(ELeave) CWsConfigChangeNotifier(aNextLevelInterface, aOwner);
222 CleanupStack::PushL(self);
228 void CWsConfigChangeNotifier::ConstructL()
230 iRetry = CEventQueueRetry::NewL();
233 void CWsConfigChangeNotifier::IssueNotificationRequest()
235 //Before submitting the request, save last configuration
236 iNextLevelInterface->GetConfiguration(iLastConfig);
237 iNextLevelInterface->NotifyOnConfigChange(iStatus);
241 void CWsConfigChangeNotifier::CancelNotificationRequest()
246 void CWsConfigChangeNotifier::UpdateLastSetConfiguration(TDisplayConfiguration& aNewConfig)
248 TSize resolution(0,0);
249 if (aNewConfig.GetResolution(resolution))
251 iLastSetConfig.SetResolution(resolution);
254 TSize resolutionTwips(0,0);
255 if (aNewConfig.GetResolutionTwips(resolutionTwips))
257 iLastSetConfig.SetResolutionTwips(resolutionTwips);
260 TDisplayConfiguration1::TRotation rotation;
261 if (aNewConfig.GetRotation(rotation))
263 iLastSetConfig.SetRotation(rotation);
267 void CWsConfigChangeNotifier::RunL()
269 if(iStatus == KErrNone)
271 //IssueNotificationRequest() will overwrite iLastConfig, save a copy first
272 TDisplayConfiguration lastConfig(iLastConfig);
273 IssueNotificationRequest();
275 TDisplayConfiguration currentConfig;
276 iNextLevelInterface->GetConfiguration(currentConfig);
277 if(!(currentConfig == lastConfig))
279 //first thing is to cancel the retry AO if it's running for the last event
280 iRetry->CancelRetry();
282 iOwner->IncreaseConfigSpinner();
284 //if the config change comes from a render stage then ensure screen device size
287 currentConfig.GetResolution(currentRes);
288 TBool disconnected = (currentRes.iHeight == 0 || currentRes.iWidth == 0) ? ETrue : EFalse;
290 //if the config change is due to CScreen::SetConfiguration() being called then we
291 //don't want to update it again. Only update if the configs are different and the
292 //display is connected...
293 TDisplayConfiguration lastSetConfig(iLastSetConfig);
294 if (!((currentConfig == lastSetConfig) || (disconnected)))
296 TDisplayConfiguration1::TRotation rotation;
297 if (lastSetConfig.GetRotation(rotation))
299 //use the latest rotation value to ensure we don't get any
300 //inconsistencies with the layer extents
301 currentConfig.SetRotation(rotation);
303 iOwner->UpdateConfiguration(currentConfig);
306 //put config change event on queue
307 RPointerArray<CWsClient> clientArray;
308 CleanupClosePushL(clientArray);
309 User::LeaveIfError(iOwner->GetNotificationClients(clientArray));
310 TBool eventOnAllQueues = ETrue;
311 for(TInt i = 0; i < clientArray.Count(); i++)
313 if(!TWindowServerEvent::SendDisplayChangedEvents(clientArray[i], iOwner->ScreenNumber(),
314 iOwner->ConfigSpinner(), iOwner->DisplaySpinner()))
316 eventOnAllQueues = EFalse;
317 clientArray[i]->SetRetryFlag(EEventDisplayChanged);
321 CleanupStack::PopAndDestroy(&clientArray);
322 //some event queues are full, kick off retry AO
323 if(!eventOnAllQueues)
325 iRetry->Init(iOwner);
326 iRetry->Retry(KRetryInitialDelay);
329 iNextLevelInterface->GetConfiguration(iLastSetConfig);
331 else if(iStatus != KErrCancel && iStatus != KErrNotSupported)
333 IssueNotificationRequest();
339 void CWsConfigChangeNotifier::DoCancel()
342 CWsConfigChangeNotifier::~CWsConfigChangeNotifier()