sl@0: /* sl@0: * Copyright (c) 2004 Nokia Corporation and/or its subsidiary(-ies). sl@0: * All rights reserved. sl@0: * This component and the accompanying materials are made available sl@0: * under the terms of "Eclipse Public License v1.0" sl@0: * which accompanies this distribution, and is available sl@0: * at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: * sl@0: * Initial Contributors: sl@0: * Nokia Corporation - initial contribution. sl@0: * sl@0: * Contributors: sl@0: * sl@0: * Description: Implementation of the Orientation effect base class sl@0: * sl@0: */ sl@0: sl@0: sl@0: sl@0: sl@0: // INCLUDE FILES sl@0: sl@0: #ifdef _DEBUG sl@0: #include sl@0: #endif sl@0: sl@0: #include sl@0: #include sl@0: #include sl@0: sl@0: sl@0: sl@0: //360 degrees: sl@0: #define TWO_PI 6283 sl@0: //180 degrees: sl@0: #define PI 3142 sl@0: //90 degrees: sl@0: #define QUARTER_PI 1570 sl@0: sl@0: sl@0: // ============================ MEMBER FUNCTIONS =============================== sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // COrientation::COrientation sl@0: // C++ default constructor can NOT contain any code, that sl@0: // might leave. sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: EXPORT_C COrientation::COrientation() sl@0: : iOrientationData(0,0,0,0,0,0,0,0,0), sl@0: iDataPckgTo(iOrientationData), sl@0: iDataPckgFrom(iOrientationData) sl@0: { sl@0: } sl@0: sl@0: // Destructor sl@0: EXPORT_C COrientation::~COrientation() sl@0: { sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // COrientation::Orientation sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: EXPORT_C void COrientation::Orientation( sl@0: TInt32& aHeading, TInt32& aPitch, TInt32& aRoll ) sl@0: { sl@0: aHeading = iOrientationData.iHeading; sl@0: aPitch = iOrientationData.iPitch; sl@0: aRoll = iOrientationData.iRoll; sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // COrientation::OrientationVectors sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: EXPORT_C void COrientation::OrientationVectors( sl@0: TInt32& aFrontX, TInt32& aFrontY, TInt32& aFrontZ, sl@0: TInt32& aAboveX, TInt32& aAboveY, TInt32& aAboveZ ) sl@0: { sl@0: aFrontX = iOrientationData.iFrontX; sl@0: aFrontY = iOrientationData.iFrontY; sl@0: aFrontZ = iOrientationData.iFrontZ; sl@0: aAboveX = iOrientationData.iAboveX; sl@0: aAboveY = iOrientationData.iAboveY; sl@0: aAboveZ = iOrientationData.iAboveZ; sl@0: sl@0: sl@0: sl@0: sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // COrientation::SetOrientationL sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: EXPORT_C void COrientation::SetOrientationL( sl@0: TInt32 aHeading, TInt32 aPitch, TInt32 aRoll ) sl@0: { sl@0: sl@0: while(aPitch > PI) sl@0: { sl@0: aPitch = aPitch - TWO_PI; sl@0: } sl@0: if(aPitch > QUARTER_PI) sl@0: { sl@0: //pitch is here between 90 and 180degs. We swap it back to between 0 and 90degs. Heading needs swaping then too. sl@0: aPitch = aPitch - (aPitch-QUARTER_PI)*2; sl@0: aHeading = aHeading + PI; sl@0: aRoll = aRoll + PI; //I'm not sure if this line is needed. Remove it if you don't get correct results for roll or upVector sl@0: } sl@0: sl@0: while(aPitch < -PI) sl@0: { sl@0: aPitch = aPitch + TWO_PI; sl@0: } sl@0: sl@0: if(aPitch < -QUARTER_PI) sl@0: { sl@0: //pitch is here between -90 and -180degs. We swap it back to between 0 and -90degs. Heading needs swaping then too. sl@0: aPitch = aPitch + (QUARTER_PI-aPitch)*2; sl@0: aHeading = aHeading + PI; sl@0: aRoll = aRoll + PI; //I'm not sure if this line is needed. Remove it if you don't get correct results for roll or upVector sl@0: } sl@0: sl@0: while (aHeading < 0) sl@0: { sl@0: aHeading = aHeading + TWO_PI; sl@0: } sl@0: sl@0: while (aHeading > TWO_PI) sl@0: { sl@0: aHeading = aHeading - TWO_PI; sl@0: } sl@0: sl@0: while (aRoll < 0) sl@0: { sl@0: aRoll = aRoll + TWO_PI; sl@0: } sl@0: while (aRoll > TWO_PI) sl@0: { sl@0: aRoll = aRoll - TWO_PI; sl@0: } sl@0: sl@0: iOrientationData.iHeading = aHeading; sl@0: iOrientationData.iPitch = aPitch; sl@0: iOrientationData.iRoll = aRoll; sl@0: sl@0: sl@0: TReal headingSin, headingCos, pitchSin, pitchCos, rollSin, rollCos; sl@0: sl@0: User::LeaveIfError( Math::Sin( headingSin, (TReal)aHeading / 1000 ) ); sl@0: User::LeaveIfError( Math::Cos( headingCos, (TReal)aHeading / 1000) ); sl@0: User::LeaveIfError( Math::Sin( pitchSin, (TReal)aPitch / 1000) ); sl@0: User::LeaveIfError( Math::Cos( pitchCos, (TReal)aPitch / 1000) ); sl@0: User::LeaveIfError( Math::Sin( rollSin, (TReal)aRoll / 1000) ); sl@0: User::LeaveIfError( Math::Cos( rollCos, (TReal)aRoll / 1000) ); sl@0: sl@0: iOrientationData.iFrontX = -headingSin * pitchCos * 1000; sl@0: iOrientationData.iFrontY = pitchSin * 1000; sl@0: iOrientationData.iFrontZ = -headingCos * pitchCos * 1000; sl@0: iOrientationData.iAboveX = (-rollSin * headingCos + rollCos * pitchSin * headingSin) * 1000; sl@0: iOrientationData.iAboveY = pitchCos * rollCos * 1000; sl@0: iOrientationData.iAboveZ = (rollSin * headingSin + rollCos * headingCos * pitchSin) * 1000; sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // COrientation::SetOrientationVectorsL sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: EXPORT_C void COrientation::SetOrientationVectorsL( sl@0: TInt32 aFrontX, TInt32 aFrontY, TInt32 aFrontZ, sl@0: TInt32 aAboveX, TInt32 aAboveY, TInt32 aAboveZ ) sl@0: { sl@0: sl@0: iOrientationData.iFrontX = aFrontX; sl@0: iOrientationData.iFrontY = aFrontY; sl@0: iOrientationData.iFrontZ = aFrontZ; sl@0: iOrientationData.iAboveX = aAboveX; sl@0: iOrientationData.iAboveY = aAboveY; sl@0: iOrientationData.iAboveZ = aAboveZ; sl@0: sl@0: TInt32 previous_valid_value_of_pitch = iOrientationData.iPitch; sl@0: sl@0: sl@0: if(!((aFrontX==0) && (aFrontZ==0))) sl@0: { sl@0: TReal xDividedByZAtan; sl@0: xDividedByZAtan = atan2 (-aFrontX, -aFrontZ); sl@0: sl@0: if(xDividedByZAtan > 0) sl@0: iOrientationData.iHeading = (TInt32) (xDividedByZAtan * 1000 + 0.5); sl@0: else sl@0: iOrientationData.iHeading = (TInt32) (xDividedByZAtan * 1000 - 0.5); sl@0: sl@0: } sl@0: sl@0: if(aFrontX == 0 && aFrontZ == 0) sl@0: { sl@0: if (aFrontY > 0) sl@0: iOrientationData.iPitch = KPi/2 * 1000 + 0.5; //verify the units! sl@0: else if (aFrontY == 0) sl@0: iOrientationData.iPitch = previous_valid_value_of_pitch; sl@0: else if (aFrontY < 0) sl@0: iOrientationData.iPitch = -KPi/2 * 1000 - 0.5; //verify the units! sl@0: } sl@0: else sl@0: { sl@0: sl@0: TReal sqrtXZ = 0, squareFx = 0, squareFz = 0; sl@0: Math::Pow(squareFx, aFrontX, 2); sl@0: Math::Pow(squareFz, aFrontZ, 2); sl@0: TReal total = squareFx + squareFz; sl@0: sl@0: Math::Sqrt (sqrtXZ, total); sl@0: sl@0: TReal yTotalAtan = atan2 ((TReal)aFrontY, sqrtXZ); sl@0: sl@0: if (yTotalAtan > 0) sl@0: iOrientationData.iPitch = (TInt32) (yTotalAtan * 1000 + 0.5); sl@0: else sl@0: iOrientationData.iPitch = (TInt32) (yTotalAtan * 1000 - 0.5); sl@0: sl@0: } sl@0: sl@0: TReal rot_aboveX = 0; sl@0: sl@0: rot_aboveX = cos(-((TReal)iOrientationData.iHeading)/1000) * (((TReal)aAboveX)/1000) + sin(-((TReal)iOrientationData.iHeading)/1000) * (((TReal)aAboveZ)/1000); sl@0: sl@0: TReal rot_aboveY = 0; sl@0: sl@0: rot_aboveY = sin(-((TReal)iOrientationData.iHeading)/1000) * sin(-((TReal)iOrientationData.iPitch)/1000) * (((TReal)aAboveX)/1000) + cos(-((TReal)iOrientationData.iPitch)/1000) * (((TReal)aAboveY)/1000) - sin(-(TReal)iOrientationData.iPitch/1000) * cos(-((TReal)iOrientationData.iHeading)/1000)* (((TReal)aAboveZ)/1000); sl@0: sl@0: TReal roll = atan2(-rot_aboveX, rot_aboveY); sl@0: sl@0: if (roll > 0) sl@0: iOrientationData.iRoll = (TInt32) (roll * 1000 + 0.5); sl@0: else sl@0: iOrientationData.iRoll = (TInt32) (roll * 1000 - 0.5); sl@0: sl@0: sl@0: while (iOrientationData.iHeading < 0) sl@0: iOrientationData.iHeading = iOrientationData.iHeading + TWO_PI; sl@0: sl@0: while (iOrientationData.iHeading > TWO_PI) sl@0: iOrientationData.iHeading = iOrientationData.iHeading - TWO_PI; sl@0: sl@0: while (iOrientationData.iPitch < 0) sl@0: iOrientationData.iPitch = iOrientationData.iPitch + TWO_PI; sl@0: sl@0: while (iOrientationData.iPitch > TWO_PI) sl@0: iOrientationData.iPitch = iOrientationData.iPitch - TWO_PI; sl@0: sl@0: while (iOrientationData.iRoll < 0) sl@0: iOrientationData.iRoll = iOrientationData.iRoll + TWO_PI; sl@0: sl@0: while (iOrientationData.iRoll > TWO_PI) sl@0: iOrientationData.iRoll = iOrientationData.iRoll - TWO_PI; sl@0: sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // COrientation::DoEffectData sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: EXPORT_C const TDesC8& COrientation::DoEffectData() sl@0: { sl@0: #ifdef _DEBUG sl@0: RDebug::Print(_L("COrientation::DoEffectData")); sl@0: #endif sl@0: iDataPckgTo = iOrientationData; sl@0: return iDataPckgTo; sl@0: } sl@0: sl@0: // ----------------------------------------------------------------------------- sl@0: // COrientation::SetEffectData sl@0: // ----------------------------------------------------------------------------- sl@0: // sl@0: EXPORT_C void COrientation::SetEffectData( sl@0: const TDesC8& aEffectDataBuffer ) sl@0: { sl@0: #ifdef _DEBUG sl@0: RDebug::Print(_L("COrientation::SetEffectData")); sl@0: #endif sl@0: TEfOrientationDataPckg dataPckg; sl@0: dataPckg.Copy(aEffectDataBuffer); sl@0: iOrientationData = dataPckg(); sl@0: iEnabled = iOrientationData.iEnabled; sl@0: iEnforced = iOrientationData.iEnforced; sl@0: iHaveUpdateRights = iOrientationData.iHaveUpdateRights; sl@0: sl@0: } sl@0: sl@0: // ========================== OTHER EXPORTED FUNCTIONS ========================= sl@0: sl@0: // End of File