sl@0: // Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This material, including documentation and any related sl@0: // computer programs, is protected by copyright controlled by sl@0: // Nokia. All rights are reserved. Copying, including sl@0: // reproducing, storing, adapting or translating, any sl@0: // or all of this material requires the prior written consent of sl@0: // Nokia. This material also contains confidential sl@0: // information which may not be disclosed to others without the sl@0: // prior written consent of Nokia. sl@0: // sl@0: // Description: sl@0: // Render Orientation Tracking and Publication sl@0: // sl@0: sl@0: #include sl@0: #include sl@0: #include "renderorientationtracker.h" sl@0: #include "rootwin.h" sl@0: #include "windowgroup.h" sl@0: #include "wstop.h" sl@0: #include "..\debuglog\DEBUGLOG.H" sl@0: sl@0: extern CDebugLogBase* wsDebugLog; sl@0: sl@0: /** Convert a TRenderOrientation value into a TDigitiserOrientation. sl@0: Note: The algorithm used makes use of the ordering of the values of the respective enums, sl@0: thus this is checked for (at compile time) at the start of the function. sl@0: @param aWservOrientation A value from the TRenderOrientation enums. sl@0: @return The equivalent value from the TDigitiserOrientation enums. sl@0: */ sl@0: inline HALData::TDigitiserOrientation WservToDigitiser(TRenderOrientation aWservOrientation) sl@0: { sl@0: __ASSERT_COMPILE(EDisplayOrientationNormal+1 == EDisplayOrientation90CW); sl@0: __ASSERT_COMPILE(EDisplayOrientationNormal+2 == EDisplayOrientation180); sl@0: __ASSERT_COMPILE(EDisplayOrientationNormal+3 == EDisplayOrientation270CW); sl@0: __ASSERT_COMPILE(HALData::EDigitiserOrientation_000+1 == HALData::EDigitiserOrientation_090); sl@0: __ASSERT_COMPILE(HALData::EDigitiserOrientation_000+2 == HALData::EDigitiserOrientation_180); sl@0: __ASSERT_COMPILE(HALData::EDigitiserOrientation_000+3 == HALData::EDigitiserOrientation_270); sl@0: HALData::TDigitiserOrientation ret=static_cast sl@0: (HALData::EDigitiserOrientation_000 + (aWservOrientation - EDisplayOrientationNormal)); sl@0: return ret; sl@0: } sl@0: sl@0: // Todo remove/undefine this for release sl@0: #define TECHVIEW_TESTMODE sl@0: sl@0: CWsRenderOrienationTracker* CWsRenderOrienationTracker::NewL() sl@0: { sl@0: CWsRenderOrienationTracker* self = new(ELeave)CWsRenderOrienationTracker(); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(); sl@0: CleanupStack::Pop(); sl@0: return self; sl@0: } sl@0: sl@0: CWsRenderOrienationTracker::CWsRenderOrienationTracker() sl@0: : CActive(CActive::EPriorityStandard), sl@0: iRenderOrientationTrackingType(EDisplayOrientationNormal), sl@0: iPublishedRenderOrientation(EDisplayOrientationNormal) sl@0: { sl@0: CActiveScheduler::Add(this); sl@0: } sl@0: sl@0: void CWsRenderOrienationTracker::ConstructL() sl@0: { sl@0: const TSecurityPolicy KRenderOrientationReadSecurityPolicy(ECapability_None); sl@0: const TSecurityPolicy KRenderOrientationWriteSecurityPolicy(ECapabilityWriteDeviceData); sl@0: sl@0: // Define P&S Property to publish to sl@0: TInt error = RProperty::Define( KRenderOrientationCategory, sl@0: KRenderOrientationKey, sl@0: RProperty::EInt, sl@0: KRenderOrientationReadSecurityPolicy, sl@0: KRenderOrientationWriteSecurityPolicy); sl@0: sl@0: // Attach the publisher for real-time publishing sl@0: if(KErrNone == error) sl@0: error = iRenderOrientationPublisher.Attach( KRenderOrientationCategory, sl@0: KRenderOrientationKey); sl@0: sl@0: // Publish the initial value sl@0: if(KErrNone == error) sl@0: error = DoPublishOrientation(EDisplayOrientationNormal); sl@0: sl@0: //Set the initial value to HAL sl@0: if(KErrNone == error) sl@0: SetHALOrientation(EDisplayOrientationNormal); sl@0: sl@0: if (wsDebugLog && KErrNone!=error) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: failed to initialise with error %d"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,logText,error); sl@0: } sl@0: User::LeaveIfError(error); sl@0: } sl@0: sl@0: CWsRenderOrienationTracker::~CWsRenderOrienationTracker() sl@0: { sl@0: Cancel(); sl@0: iRenderOrientationPublisher.Delete(KRenderOrientationCategory, KRenderOrientationKey); sl@0: iRenderOrientationPublisher.Close(); sl@0: } sl@0: sl@0: /** sl@0: If the orientation of the given window group is useable updates aOrientationTrackingType with the orientation sl@0: sl@0: @param Input: the window group to check sl@0: @param Output: the window group's orientation if usable ( otherwise unchanged ) sl@0: @return KErrNone if the orienation is usable, KErrNotFound if the orientation is not useable, KErrNotSupported if the orientation is unknown sl@0: */ sl@0: TInt CWsRenderOrienationTracker::CheckWindowGroupOrientation(const CWsWindowGroup& aWinGroup, TRenderOrientationTrackingType& aOrientationTrackingType) sl@0: { sl@0: TInt error = KErrNone; sl@0: TRenderOrientationTrackingType tempOrientationTrackingType = static_cast(aWinGroup.WsOwner()->GetIndicatedAppOrientation()); sl@0: switch(tempOrientationTrackingType) sl@0: { sl@0: case EDisplayOrientationNormal: sl@0: case EDisplayOrientation90CW: sl@0: case EDisplayOrientation180: sl@0: case EDisplayOrientation270CW: sl@0: case EDisplayOrientationAuto: sl@0: aOrientationTrackingType = tempOrientationTrackingType; sl@0: break; sl@0: sl@0: case EDisplayOrientationIgnore: sl@0: error = KErrNotFound; sl@0: if (wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: winGroup %08x orientation is set to be ignored"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText,reinterpret_cast(&aWinGroup)); sl@0: } sl@0: break; sl@0: sl@0: default: sl@0: error = KErrNotSupported; sl@0: if (wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: winGroup %08x has undefined orientation, Error %d"); sl@0: TBuf buf; sl@0: buf.Format(logText, &aWinGroup, error); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,buf); sl@0: } sl@0: break; sl@0: } sl@0: sl@0: return error; sl@0: } sl@0: sl@0: /** sl@0: Checks that the given group window is appropriate for dictating the render orientation sl@0: sl@0: @param Input: The group window to check sl@0: @return ETrue is the group window is usable, else EFalse sl@0: */ sl@0: TBool CWsRenderOrienationTracker::UseableGroupWindow(const CWsWindowGroup& aWinGroup) const sl@0: { sl@0: #ifdef TECHVIEW_TESTMODE sl@0: // for some reason IsFocusable seems to return 0 and 2, not 0 and 1 sl@0: return NULL!=aWinGroup.Child() && sl@0: (aWinGroup.IsFocusable() ? ETrue : EFalse); sl@0: #else sl@0: return (NULL!=aWinGroup.Child()); sl@0: #endif sl@0: } sl@0: sl@0: /** sl@0: Finds the topmost usable windowgroup which has a usable orientation, and outputs that orientation sl@0: sl@0: @param Output: The current render orientation sl@0: @return KErrNone if successful, KErrNotFound if the focus window group is not usable, KErrNotSupported if an invalid orientation is found sl@0: */ sl@0: TInt CWsRenderOrienationTracker::GetFocusWindowOrientation(TRenderOrientationTrackingType& aOrientationTrackingType) sl@0: { sl@0: TInt error = KErrNone; sl@0: CWsWindowGroup* focusWinGroup = CWsTop::FocusWindowGroup(); sl@0: if(!focusWinGroup) sl@0: { sl@0: if(wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: focusWinGroup not found"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText); sl@0: } sl@0: error = KErrNotFound; sl@0: } sl@0: else sl@0: { sl@0: error = CheckWindowGroupOrientation(*focusWinGroup, aOrientationTrackingType); sl@0: } sl@0: return error; sl@0: } sl@0: sl@0: /** sl@0: Finds the topmost usable windowgroup which has a usable orientation, and outputs that orientation sl@0: sl@0: @param Output: The current render orientation sl@0: @return KErrNone if successful, KErrNotSupported if an invalid orientation is found sl@0: */ sl@0: TInt CWsRenderOrienationTracker::FindOrientationFromWindowTree(TRenderOrientationTrackingType& aOrientationTrackingType) sl@0: { sl@0: TInt error = KErrNone; sl@0: TRenderOrientationTrackingType tempOrientationTrackingType = iRenderOrientationTrackingType; sl@0: CWsRootWindow* rootWin = CWsTop::CurrentFocusScreen()->RootWindow(); sl@0: TBool finished = EFalse; sl@0: for(CWsWindowGroup* winGroup = rootWin->Child(); !finished && NULL != winGroup; winGroup = winGroup->NextSibling()) sl@0: { sl@0: if (wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: winGroup %08x has priority %d, Orientation %d, Focusable %d, Child %08x"); sl@0: TBuf buf; sl@0: buf.Format(logText, winGroup, winGroup->OrdinalPriority(), winGroup->WsOwner()->GetIndicatedAppOrientation(), sl@0: winGroup->IsFocusable()?ETrue:EFalse, winGroup->Child()); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,buf); sl@0: } sl@0: // winGroup is a higher priority ordinal, so see if it has an orientation that can be used sl@0: // although we're only interested in window groups with child windows otherwise nothing is visible anyway sl@0: if(UseableGroupWindow(*winGroup)) sl@0: { sl@0: error = CheckWindowGroupOrientation(*winGroup, tempOrientationTrackingType); sl@0: switch(error) sl@0: { sl@0: case KErrNone: sl@0: { sl@0: // List is in order, so just find the first one sl@0: if (wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Found winGroup %08x with Orientation %d"); sl@0: TBuf buf; sl@0: buf.Format(logText, winGroup, winGroup->WsOwner()->GetIndicatedAppOrientation()); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,buf); sl@0: } sl@0: finished = ETrue; sl@0: break; sl@0: } sl@0: sl@0: case KErrNotFound: sl@0: // so keep searching sl@0: break; sl@0: sl@0: case KErrNotSupported: sl@0: default: sl@0: finished = ETrue; sl@0: break; sl@0: sl@0: } sl@0: sl@0: } sl@0: } sl@0: // Safe even in error code as won't have been changed by CheckWindowGroupOrientation sl@0: aOrientationTrackingType = tempOrientationTrackingType; sl@0: sl@0: return error; sl@0: } sl@0: sl@0: /** sl@0: First checks to see if the focus window group has a usable orientation, if so that is output. sl@0: Otherwise, finds the topmost usable windowgroup which has a usable orientation, and outputs that sl@0: sl@0: @param Output: The current render orientation sl@0: @return KErrNone if successful, KErrNotSupported if an invalid orientation is found sl@0: */ sl@0: TInt CWsRenderOrienationTracker::GetIndicatedOrientation(TRenderOrientationTrackingType& aOrientationTrackingType) sl@0: { sl@0: // First check the focus window group sl@0: TInt error = GetFocusWindowOrientation(aOrientationTrackingType); sl@0: sl@0: // Don't look for another window if the focus window is usable sl@0: // or if an error has occured, then don't change current orientation sl@0: switch(error) sl@0: { sl@0: case KErrNone: sl@0: { sl@0: if(wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Using focus window %08x for orientation"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText,reinterpret_cast(CWsTop::FocusWindowGroup())); sl@0: } sl@0: break; sl@0: } sl@0: sl@0: case KErrNotFound: sl@0: { sl@0: // Can't use focus window group, so find the topmost windowgroup with a valid orientation sl@0: error = FindOrientationFromWindowTree(aOrientationTrackingType); sl@0: break; sl@0: } sl@0: sl@0: default: sl@0: // Unrecoverable error, abort and leave published orientation unchanged sl@0: break; sl@0: } sl@0: sl@0: return error; sl@0: } sl@0: sl@0: /** sl@0: Checks to see if the render orientation has changed, and publishes any new orientaion sl@0: via publish and subscribe sl@0: sl@0: @see KRenderOrientationCategory sl@0: @see KRenderOrientationKey sl@0: */ sl@0: void CWsRenderOrienationTracker::CheckRenderOrientation() sl@0: { sl@0: TRenderOrientationTrackingType newOrientationTrackingType = iRenderOrientationTrackingType; sl@0: TInt error = GetIndicatedOrientation(newOrientationTrackingType); sl@0: sl@0: // if the tracking type has changed... sl@0: if(KErrNone == error && iRenderOrientationTrackingType != newOrientationTrackingType) sl@0: { sl@0: if(EDisplayOrientationAuto == iRenderOrientationTrackingType) sl@0: { sl@0: // change from auto type, so we need to cancel request for updates from the theme server sl@0: Cancel(); sl@0: } sl@0: iRenderOrientationTrackingType = newOrientationTrackingType; sl@0: if(EDisplayOrientationAuto == iRenderOrientationTrackingType) sl@0: { sl@0: // Change to auto type, so we need to request updates from the theme server sl@0: // Attach to the Theme server to get orientation change updates sl@0: error = iThemeOrientationProperty.Attach( KThemeOrientationCategory, KThemeOrientationKey ); sl@0: if (wsDebugLog) sl@0: { sl@0: if(KErrNone == error) sl@0: { sl@0: // Information Log sl@0: _LIT(logText,"Orientation Tracker: Attached to theme orientation property"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText); sl@0: } sl@0: else sl@0: { sl@0: // Error Log sl@0: _LIT(logText,"Orientation Tracker: Error %d attaching to theme orientation property"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,logText, error); sl@0: } sl@0: } sl@0: sl@0: RequestDeviceOrientationNotification(); sl@0: } sl@0: // See if the has changed, and publish if it has sl@0: error = DoOrientationTracking(); sl@0: } sl@0: sl@0: if (wsDebugLog && KErrNone != error) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Error %d Checking Render Orientation"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,logText, error); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Requests notification of change of the theme server orientation sl@0: sl@0: @Pre iThemeOrientationProperty has had Attach called on it sl@0: */ sl@0: void CWsRenderOrienationTracker::RequestDeviceOrientationNotification() sl@0: { sl@0: if(!IsActive()) sl@0: { sl@0: if (wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Subscribing to theme orientation property"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText); sl@0: } sl@0: // Request for Theme Server Orientation P&S sl@0: iThemeOrientationProperty.Subscribe(iStatus); sl@0: SetActive(); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Cancels and closes (detaches) from the theme orientation publish and subscribe sl@0: */ sl@0: void CWsRenderOrienationTracker::CancelDeviceOrientationNotification() sl@0: { sl@0: // Cancel Request for Theme Server Orientation P&S sl@0: iThemeOrientationProperty.Cancel(); sl@0: iThemeOrientationProperty.Close(); sl@0: sl@0: if (wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Cancelled/closed theme orientation property"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything,logText); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Called when the theme servers orientation has changed. sl@0: Re-requests unless cancelled sl@0: */ sl@0: void CWsRenderOrienationTracker::RunL() sl@0: { sl@0: TInt error = iStatus.Int(); sl@0: if(KErrNone == error) sl@0: { sl@0: // Re-request sl@0: RequestDeviceOrientationNotification(); sl@0: sl@0: TInt error = DoOrientationTracking(); sl@0: if (wsDebugLog && KErrNone != error) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Error %d processing theme orientation property"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,logText, error); sl@0: } sl@0: } sl@0: else if (wsDebugLog && KErrCancel != error) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Error %d from theme orientation property, not resubscribed"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,logText, error); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Cancels the request for notification for changes to theme orientation sl@0: */ sl@0: void CWsRenderOrienationTracker::DoCancel() sl@0: { sl@0: CancelDeviceOrientationNotification(); sl@0: } sl@0: sl@0: /** sl@0: Gets the orientation published from theme server sl@0: sl@0: @param Output: the theme server orientation sl@0: @return KErrNone if successful, KErrNotSupported if the theme server returns an unknown orientation, else any of the system wide error codes sl@0: */ sl@0: TInt CWsRenderOrienationTracker::GetThemeOrientation(TRenderOrientation& aThemeOrientation) sl@0: { sl@0: TInt themeOrientation=EDisplayOrientationNormal; sl@0: TInt error = iThemeOrientationProperty.Get(themeOrientation); sl@0: if(wsDebugLog && KErrNone != error) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Error %d getting theme orientation property"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,logText, error); sl@0: } sl@0: sl@0: if(KErrNone == error) sl@0: { sl@0: // Translate the received orientation sl@0: switch(themeOrientation) sl@0: { sl@0: case EDisplayOrientationNormal: sl@0: case EDisplayOrientation90CW: sl@0: case EDisplayOrientation180: sl@0: case EDisplayOrientation270CW: sl@0: // only update if orientation is supported sl@0: aThemeOrientation = static_cast(themeOrientation); sl@0: break; sl@0: sl@0: default: sl@0: error = KErrNotSupported; sl@0: if (wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Unsupported orientation %d from theme orientation property, Error %d"); sl@0: TBuf buf; sl@0: buf.Format(logText, themeOrientation, error); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,buf); sl@0: } sl@0: break; sl@0: } sl@0: } sl@0: return error; sl@0: } sl@0: sl@0: /** sl@0: Processes the indicated orientation into an actual orientation sl@0: sl@0: @return KErrNone for success, KErrNotSupported if the orientation is unknown, else any of the system wide error codes sl@0: */ sl@0: TInt CWsRenderOrienationTracker::DoOrientationTracking() sl@0: { sl@0: TInt error = KErrNone; sl@0: TRenderOrientation newDeviceOrientation; sl@0: switch(iRenderOrientationTrackingType) sl@0: { sl@0: case EDisplayOrientationNormal: sl@0: case EDisplayOrientation90CW: sl@0: case EDisplayOrientation180: sl@0: case EDisplayOrientation270CW: sl@0: newDeviceOrientation = iRenderOrientationTrackingType; sl@0: break; sl@0: sl@0: case EDisplayOrientationAuto: sl@0: error = GetThemeOrientation(newDeviceOrientation); sl@0: break; sl@0: sl@0: default: sl@0: error = KErrNotSupported; sl@0: if (wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Unsupported orientation tracking type %d, error %d"); sl@0: TBuf buf; sl@0: buf.Format(logText, iRenderOrientationTrackingType, error); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate,buf); sl@0: } sl@0: break; sl@0: } sl@0: sl@0: if(KErrNone == error) sl@0: { sl@0: error = PublishOrientation(newDeviceOrientation); sl@0: } sl@0: sl@0: return error; sl@0: } sl@0: sl@0: /** sl@0: Publishes the given value sl@0: sl@0: @param The render orientation to publish sl@0: @return KErrNone for success, else any of the system wide erro codes sl@0: */ sl@0: TInt CWsRenderOrienationTracker::DoPublishOrientation(const TRenderOrientation aRenderOrientation) sl@0: { sl@0: TInt error = iRenderOrientationPublisher.Set(aRenderOrientation); sl@0: sl@0: // if it's published OK, then remember the newly published value sl@0: if(KErrNone == error) sl@0: { sl@0: iPublishedRenderOrientation = aRenderOrientation; sl@0: if(wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Published render orientation %d"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, logText, aRenderOrientation); sl@0: } sl@0: } sl@0: else if(wsDebugLog) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Error %d setting render orientation property"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate, logText, error); sl@0: } sl@0: return error; sl@0: } sl@0: sl@0: void CWsRenderOrienationTracker::SetHALOrientation(const TRenderOrientation aRenderOrientation) sl@0: { sl@0: // If the render orientation is EDisplayOrientationAuto then don't update HAL sl@0: // The application and HAL should always have the same state for the orientation. sl@0: if(EDisplayOrientationAuto != aRenderOrientation) sl@0: { sl@0: TInt error = HAL::Set(CWsTop::CurrentFocusScreen()->ScreenNumber(), HALData::EDigitiserOrientation, WservToDigitiser(iPublishedRenderOrientation)); sl@0: //Just log the error if there is one. sl@0: if(wsDebugLog && error != KErrNone) sl@0: { sl@0: _LIT(logText,"Orientation Tracker: Error %d setting digitiser orientation"); sl@0: wsDebugLog->MiscMessage(CDebugLogBase::ELogIntermediate, logText, error); sl@0: } sl@0: } sl@0: } sl@0: sl@0: /** sl@0: If the current orientation differs from the previously published value then publishes the current value sl@0: sl@0: @param The render orientation to check and publish sl@0: @return KErrNone for success, else any of the system wide erro codes sl@0: */ sl@0: TInt CWsRenderOrienationTracker::PublishOrientation(const TRenderOrientation aRenderOrientation) sl@0: { sl@0: TInt error = KErrNone; sl@0: sl@0: if(aRenderOrientation != iPublishedRenderOrientation) sl@0: { sl@0: // If the device Orientation has changed, publish it sl@0: error = DoPublishOrientation(aRenderOrientation); sl@0: if(KErrNone == error) sl@0: SetHALOrientation(aRenderOrientation); sl@0: } sl@0: return error; sl@0: } sl@0: