First public contribution.
1 // Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This material, including documentation and any related
4 // computer programs, is protected by copyright controlled by
5 // Nokia. All rights are reserved. Copying, including
6 // reproducing, storing, adapting or translating, any
7 // or all of this material requires the prior written consent of
8 // Nokia. This material also contains confidential
9 // information which may not be disclosed to others without the
10 // prior written consent of Nokia.
13 // Render Orientation Tracking and Publication
18 #include "renderorientationtracker.h"
20 #include "windowgroup.h"
22 #include "..\debuglog\DEBUGLOG.H"
24 extern CDebugLogBase* wsDebugLog;
26 /** Convert a TRenderOrientation value into a TDigitiserOrientation.
27 Note: The algorithm used makes use of the ordering of the values of the respective enums,
28 thus this is checked for (at compile time) at the start of the function.
29 @param aWservOrientation A value from the TRenderOrientation enums.
30 @return The equivalent value from the TDigitiserOrientation enums.
32 inline HALData::TDigitiserOrientation WservToDigitiser(TRenderOrientation aWservOrientation)
34 __ASSERT_COMPILE(EDisplayOrientationNormal+1 == EDisplayOrientation90CW);
35 __ASSERT_COMPILE(EDisplayOrientationNormal+2 == EDisplayOrientation180);
36 __ASSERT_COMPILE(EDisplayOrientationNormal+3 == EDisplayOrientation270CW);
37 __ASSERT_COMPILE(HALData::EDigitiserOrientation_000+1 == HALData::EDigitiserOrientation_090);
38 __ASSERT_COMPILE(HALData::EDigitiserOrientation_000+2 == HALData::EDigitiserOrientation_180);
39 __ASSERT_COMPILE(HALData::EDigitiserOrientation_000+3 == HALData::EDigitiserOrientation_270);
40 HALData::TDigitiserOrientation ret=static_cast<HALData::TDigitiserOrientation>
41 (HALData::EDigitiserOrientation_000 + (aWservOrientation - EDisplayOrientationNormal));
45 // Todo remove/undefine this for release
46 #define TECHVIEW_TESTMODE
48 CWsRenderOrienationTracker* CWsRenderOrienationTracker::NewL()
50 CWsRenderOrienationTracker* self = new(ELeave)CWsRenderOrienationTracker();
51 CleanupStack::PushL(self);
57 CWsRenderOrienationTracker::CWsRenderOrienationTracker()
58 : CActive(CActive::EPriorityStandard),
59 iRenderOrientationTrackingType(EDisplayOrientationNormal),
60 iPublishedRenderOrientation(EDisplayOrientationNormal)
62 CActiveScheduler::Add(this);
65 void CWsRenderOrienationTracker::ConstructL()
67 const TSecurityPolicy KRenderOrientationReadSecurityPolicy(ECapability_None);
68 const TSecurityPolicy KRenderOrientationWriteSecurityPolicy(ECapabilityWriteDeviceData);
70 // Define P&S Property to publish to
71 TInt error = RProperty::Define( KRenderOrientationCategory,
72 KRenderOrientationKey,
74 KRenderOrientationReadSecurityPolicy,
75 KRenderOrientationWriteSecurityPolicy);
77 // Attach the publisher for real-time publishing
79 error = iRenderOrientationPublisher.Attach( KRenderOrientationCategory,
80 KRenderOrientationKey);
82 // Publish the initial value
84 error = DoPublishOrientation(EDisplayOrientationNormal);
86 //Set the initial value to HAL
88 SetHALOrientation(EDisplayOrientationNormal);
90 if (wsDebugLog && KErrNone!=error)
92 _LIT(logText,"Orientation Tracker: failed to initialise with error %d");
93 wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,logText,error);
95 User::LeaveIfError(error);
98 CWsRenderOrienationTracker::~CWsRenderOrienationTracker()
101 iRenderOrientationPublisher.Delete(KRenderOrientationCategory, KRenderOrientationKey);
102 iRenderOrientationPublisher.Close();
106 If the orientation of the given window group is useable updates aOrientationTrackingType with the orientation
108 @param Input: the window group to check
109 @param Output: the window group's orientation if usable ( otherwise unchanged )
110 @return KErrNone if the orienation is usable, KErrNotFound if the orientation is not useable, KErrNotSupported if the orientation is unknown
112 TInt CWsRenderOrienationTracker::CheckWindowGroupOrientation(const CWsWindowGroup& aWinGroup, TRenderOrientationTrackingType& aOrientationTrackingType)
114 TInt error = KErrNone;
115 TRenderOrientationTrackingType tempOrientationTrackingType = static_cast<TRenderOrientationTrackingType>(aWinGroup.WsOwner()->GetIndicatedAppOrientation());
116 switch(tempOrientationTrackingType)
118 case EDisplayOrientationNormal:
119 case EDisplayOrientation90CW:
120 case EDisplayOrientation180:
121 case EDisplayOrientation270CW:
122 case EDisplayOrientationAuto:
123 aOrientationTrackingType = tempOrientationTrackingType;
126 case EDisplayOrientationIgnore:
127 error = KErrNotFound;
130 _LIT(logText,"Orientation Tracker: winGroup %08x orientation is set to be ignored");
131 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText,reinterpret_cast<TInt>(&aWinGroup));
136 error = KErrNotSupported;
139 _LIT(logText,"Orientation Tracker: winGroup %08x has undefined orientation, Error %d");
140 TBuf<LogTBufSize> buf;
141 buf.Format(logText, &aWinGroup, error);
142 wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,buf);
151 Checks that the given group window is appropriate for dictating the render orientation
153 @param Input: The group window to check
154 @return ETrue is the group window is usable, else EFalse
156 TBool CWsRenderOrienationTracker::UseableGroupWindow(const CWsWindowGroup& aWinGroup) const
158 #ifdef TECHVIEW_TESTMODE
159 // for some reason IsFocusable seems to return 0 and 2, not 0 and 1
160 return NULL!=aWinGroup.Child() &&
161 (aWinGroup.IsFocusable() ? ETrue : EFalse);
163 return (NULL!=aWinGroup.Child());
168 Finds the topmost usable windowgroup which has a usable orientation, and outputs that orientation
170 @param Output: The current render orientation
171 @return KErrNone if successful, KErrNotFound if the focus window group is not usable, KErrNotSupported if an invalid orientation is found
173 TInt CWsRenderOrienationTracker::GetFocusWindowOrientation(TRenderOrientationTrackingType& aOrientationTrackingType)
175 TInt error = KErrNone;
176 CWsWindowGroup* focusWinGroup = CWsTop::FocusWindowGroup();
181 _LIT(logText,"Orientation Tracker: focusWinGroup not found");
182 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText);
184 error = KErrNotFound;
188 error = CheckWindowGroupOrientation(*focusWinGroup, aOrientationTrackingType);
194 Finds the topmost usable windowgroup which has a usable orientation, and outputs that orientation
196 @param Output: The current render orientation
197 @return KErrNone if successful, KErrNotSupported if an invalid orientation is found
199 TInt CWsRenderOrienationTracker::FindOrientationFromWindowTree(TRenderOrientationTrackingType& aOrientationTrackingType)
201 TInt error = KErrNone;
202 TRenderOrientationTrackingType tempOrientationTrackingType = iRenderOrientationTrackingType;
203 CWsRootWindow* rootWin = CWsTop::CurrentFocusScreen()->RootWindow();
204 TBool finished = EFalse;
205 for(CWsWindowGroup* winGroup = rootWin->Child(); !finished && NULL != winGroup; winGroup = winGroup->NextSibling())
209 _LIT(logText,"Orientation Tracker: winGroup %08x has priority %d, Orientation %d, Focusable %d, Child %08x");
210 TBuf<LogTBufSize> buf;
211 buf.Format(logText, winGroup, winGroup->OrdinalPriority(), winGroup->WsOwner()->GetIndicatedAppOrientation(),
212 winGroup->IsFocusable()?ETrue:EFalse, winGroup->Child());
213 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,buf);
215 // winGroup is a higher priority ordinal, so see if it has an orientation that can be used
216 // although we're only interested in window groups with child windows otherwise nothing is visible anyway
217 if(UseableGroupWindow(*winGroup))
219 error = CheckWindowGroupOrientation(*winGroup, tempOrientationTrackingType);
224 // List is in order, so just find the first one
227 _LIT(logText,"Orientation Tracker: Found winGroup %08x with Orientation %d");
228 TBuf<LogTBufSize> buf;
229 buf.Format(logText, winGroup, winGroup->WsOwner()->GetIndicatedAppOrientation());
230 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,buf);
240 case KErrNotSupported:
249 // Safe even in error code as won't have been changed by CheckWindowGroupOrientation
250 aOrientationTrackingType = tempOrientationTrackingType;
256 First checks to see if the focus window group has a usable orientation, if so that is output.
257 Otherwise, finds the topmost usable windowgroup which has a usable orientation, and outputs that
259 @param Output: The current render orientation
260 @return KErrNone if successful, KErrNotSupported if an invalid orientation is found
262 TInt CWsRenderOrienationTracker::GetIndicatedOrientation(TRenderOrientationTrackingType& aOrientationTrackingType)
264 // First check the focus window group
265 TInt error = GetFocusWindowOrientation(aOrientationTrackingType);
267 // Don't look for another window if the focus window is usable
268 // or if an error has occured, then don't change current orientation
275 _LIT(logText,"Orientation Tracker: Using focus window %08x for orientation");
276 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText,reinterpret_cast<TInt>(CWsTop::FocusWindowGroup()));
283 // Can't use focus window group, so find the topmost windowgroup with a valid orientation
284 error = FindOrientationFromWindowTree(aOrientationTrackingType);
289 // Unrecoverable error, abort and leave published orientation unchanged
297 Checks to see if the render orientation has changed, and publishes any new orientaion
298 via publish and subscribe
300 @see KRenderOrientationCategory
301 @see KRenderOrientationKey
303 void CWsRenderOrienationTracker::CheckRenderOrientation()
305 TRenderOrientationTrackingType newOrientationTrackingType = iRenderOrientationTrackingType;
306 TInt error = GetIndicatedOrientation(newOrientationTrackingType);
308 // if the tracking type has changed...
309 if(KErrNone == error && iRenderOrientationTrackingType != newOrientationTrackingType)
311 if(EDisplayOrientationAuto == iRenderOrientationTrackingType)
313 // change from auto type, so we need to cancel request for updates from the theme server
316 iRenderOrientationTrackingType = newOrientationTrackingType;
317 if(EDisplayOrientationAuto == iRenderOrientationTrackingType)
319 // Change to auto type, so we need to request updates from the theme server
320 // Attach to the Theme server to get orientation change updates
321 error = iThemeOrientationProperty.Attach( KThemeOrientationCategory, KThemeOrientationKey );
324 if(KErrNone == error)
327 _LIT(logText,"Orientation Tracker: Attached to theme orientation property");
328 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText);
333 _LIT(logText,"Orientation Tracker: Error %d attaching to theme orientation property");
334 wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,logText, error);
338 RequestDeviceOrientationNotification();
340 // See if the has changed, and publish if it has
341 error = DoOrientationTracking();
344 if (wsDebugLog && KErrNone != error)
346 _LIT(logText,"Orientation Tracker: Error %d Checking Render Orientation");
347 wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,logText, error);
352 Requests notification of change of the theme server orientation
354 @Pre iThemeOrientationProperty has had Attach called on it
356 void CWsRenderOrienationTracker::RequestDeviceOrientationNotification()
362 _LIT(logText,"Orientation Tracker: Subscribing to theme orientation property");
363 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText);
365 // Request for Theme Server Orientation P&S
366 iThemeOrientationProperty.Subscribe(iStatus);
372 Cancels and closes (detaches) from the theme orientation publish and subscribe
374 void CWsRenderOrienationTracker::CancelDeviceOrientationNotification()
376 // Cancel Request for Theme Server Orientation P&S
377 iThemeOrientationProperty.Cancel();
378 iThemeOrientationProperty.Close();
382 _LIT(logText,"Orientation Tracker: Cancelled/closed theme orientation property");
383 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText);
388 Called when the theme servers orientation has changed.
389 Re-requests unless cancelled
391 void CWsRenderOrienationTracker::RunL()
393 TInt error = iStatus.Int();
394 if(KErrNone == error)
397 RequestDeviceOrientationNotification();
399 TInt error = DoOrientationTracking();
400 if (wsDebugLog && KErrNone != error)
402 _LIT(logText,"Orientation Tracker: Error %d processing theme orientation property");
403 wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,logText, error);
406 else if (wsDebugLog && KErrCancel != error)
408 _LIT(logText,"Orientation Tracker: Error %d from theme orientation property, not resubscribed");
409 wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,logText, error);
414 Cancels the request for notification for changes to theme orientation
416 void CWsRenderOrienationTracker::DoCancel()
418 CancelDeviceOrientationNotification();
422 Gets the orientation published from theme server
424 @param Output: the theme server orientation
425 @return KErrNone if successful, KErrNotSupported if the theme server returns an unknown orientation, else any of the system wide error codes
427 TInt CWsRenderOrienationTracker::GetThemeOrientation(TRenderOrientation& aThemeOrientation)
429 TInt themeOrientation=EDisplayOrientationNormal;
430 TInt error = iThemeOrientationProperty.Get(themeOrientation);
431 if(wsDebugLog && KErrNone != error)
433 _LIT(logText,"Orientation Tracker: Error %d getting theme orientation property");
434 wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,logText, error);
437 if(KErrNone == error)
439 // Translate the received orientation
440 switch(themeOrientation)
442 case EDisplayOrientationNormal:
443 case EDisplayOrientation90CW:
444 case EDisplayOrientation180:
445 case EDisplayOrientation270CW:
446 // only update if orientation is supported
447 aThemeOrientation = static_cast<TRenderOrientation>(themeOrientation);
451 error = KErrNotSupported;
454 _LIT(logText,"Orientation Tracker: Unsupported orientation %d from theme orientation property, Error %d");
455 TBuf<LogTBufSize> buf;
456 buf.Format(logText, themeOrientation, error);
457 wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,buf);
466 Processes the indicated orientation into an actual orientation
468 @return KErrNone for success, KErrNotSupported if the orientation is unknown, else any of the system wide error codes
470 TInt CWsRenderOrienationTracker::DoOrientationTracking()
472 TInt error = KErrNone;
473 TRenderOrientation newDeviceOrientation;
474 switch(iRenderOrientationTrackingType)
476 case EDisplayOrientationNormal:
477 case EDisplayOrientation90CW:
478 case EDisplayOrientation180:
479 case EDisplayOrientation270CW:
480 newDeviceOrientation = iRenderOrientationTrackingType;
483 case EDisplayOrientationAuto:
484 error = GetThemeOrientation(newDeviceOrientation);
488 error = KErrNotSupported;
491 _LIT(logText,"Orientation Tracker: Unsupported orientation tracking type %d, error %d");
492 TBuf<LogTBufSize> buf;
493 buf.Format(logText, iRenderOrientationTrackingType, error);
494 wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,buf);
499 if(KErrNone == error)
501 error = PublishOrientation(newDeviceOrientation);
508 Publishes the given value
510 @param The render orientation to publish
511 @return KErrNone for success, else any of the system wide erro codes
513 TInt CWsRenderOrienationTracker::DoPublishOrientation(const TRenderOrientation aRenderOrientation)
515 TInt error = iRenderOrientationPublisher.Set(aRenderOrientation);
517 // if it's published OK, then remember the newly published value
518 if(KErrNone == error)
520 iPublishedRenderOrientation = aRenderOrientation;
523 _LIT(logText,"Orientation Tracker: Published render orientation %d");
524 wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, logText, aRenderOrientation);
529 _LIT(logText,"Orientation Tracker: Error %d setting render orientation property");
530 wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate, logText, error);
535 void CWsRenderOrienationTracker::SetHALOrientation(const TRenderOrientation aRenderOrientation)
537 // If the render orientation is EDisplayOrientationAuto then don't update HAL
538 // The application and HAL should always have the same state for the orientation.
539 if(EDisplayOrientationAuto != aRenderOrientation)
541 TInt error = HAL::Set(CWsTop::CurrentFocusScreen()->ScreenNumber(), HALData::EDigitiserOrientation, WservToDigitiser(iPublishedRenderOrientation));
542 //Just log the error if there is one.
543 if(wsDebugLog && error != KErrNone)
545 _LIT(logText,"Orientation Tracker: Error %d setting digitiser orientation");
546 wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate, logText, error);
552 If the current orientation differs from the previously published value then publishes the current value
554 @param The render orientation to check and publish
555 @return KErrNone for success, else any of the system wide erro codes
557 TInt CWsRenderOrienationTracker::PublishOrientation(const TRenderOrientation aRenderOrientation)
559 TInt error = KErrNone;
561 if(aRenderOrientation != iPublishedRenderOrientation)
563 // If the device Orientation has changed, publish it
564 error = DoPublishOrientation(aRenderOrientation);
565 if(KErrNone == error)
566 SetHALOrientation(aRenderOrientation);